From 125172edac6479e9663a7b873577036fed2c58d0 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" Date: Fri, 15 Nov 2024 00:06:20 +0000 Subject: [PATCH] Added chart versions: buoyant/linkerd-control-plane: - 2024.11.3 buoyant/linkerd-crds: - 2024.11.3 instana/instana-agent: - 2.0.2 percona/psmdb-db: - 1.18.0 percona/psmdb-operator: - 1.18.0 redpanda/redpanda: - 5.9.10 --- .../linkerd-control-plane-2024.11.2.tgz | Bin 33401 -> 33396 bytes .../linkerd-control-plane-2024.11.3.tgz | Bin 0 -> 33442 bytes assets/buoyant/linkerd-crds-2024.11.3.tgz | Bin 0 -> 126773 bytes assets/instana/instana-agent-2.0.2.tgz | Bin 0 -> 116586 bytes assets/percona/psmdb-db-1.18.0.tgz | Bin 0 -> 16217 bytes assets/percona/psmdb-operator-1.18.0.tgz | Bin 0 -> 52509 bytes assets/redpanda/redpanda-5.9.10.tgz | Bin 0 -> 138864 bytes .../2024.11.2/Chart.yaml | 1 - .../2024.11.3/.helmignore | 22 + .../2024.11.3/Chart.lock | 6 + .../2024.11.3/Chart.yaml | 29 + .../linkerd-control-plane/2024.11.3/README.md | 322 + .../2024.11.3/README.md.gotmpl | 133 + .../2024.11.3/app-readme.md | 14 + .../2024.11.3/charts/partials/.helmignore | 21 + .../2024.11.3/charts/partials/Chart.yaml | 5 + .../2024.11.3/charts/partials/README.md | 9 + .../charts/partials/README.md.gotmpl | 14 + .../charts/partials/templates/NOTES.txt | 0 .../charts/partials/templates/_affinity.tpl | 38 + .../partials/templates/_capabilities.tpl | 16 + .../charts/partials/templates/_debug.tpl | 15 + .../charts/partials/templates/_helpers.tpl | 14 + .../charts/partials/templates/_metadata.tpl | 17 + .../partials/templates/_network-validator.tpl | 45 + .../partials/templates/_nodeselector.tpl | 4 + .../partials/templates/_proxy-config-ann.tpl | 18 + .../charts/partials/templates/_proxy-init.tpl | 98 + .../charts/partials/templates/_proxy.tpl | 271 + .../partials/templates/_pull-secrets.tpl | 6 + .../charts/partials/templates/_resources.tpl | 28 + .../partials/templates/_tolerations.tpl | 4 + .../charts/partials/templates/_trace.tpl | 5 + .../charts/partials/templates/_validate.tpl | 19 + .../charts/partials/templates/_volumes.tpl | 41 + .../2024.11.3/charts/partials/values.yaml | 0 .../2024.11.3/questions.yaml | 19 + .../2024.11.3/templates/NOTES.txt | 19 + .../2024.11.3/templates/config-rbac.yaml | 16 + .../2024.11.3/templates/config.yaml | 39 + .../2024.11.3/templates/destination-rbac.yaml | 339 + .../2024.11.3/templates/destination.yaml | 448 + .../2024.11.3/templates/heartbeat-rbac.yaml | 78 + .../2024.11.3/templates/heartbeat.yaml | 101 + .../2024.11.3/templates/identity-rbac.yaml | 49 + .../2024.11.3/templates/identity.yaml | 277 + .../2024.11.3/templates/namespace.yaml | 18 + .../2024.11.3/templates/podmonitor.yaml | 128 + .../templates/proxy-injector-rbac.yaml | 120 + .../2024.11.3/templates/proxy-injector.yaml | 227 + .../2024.11.3/templates/psp.yaml | 119 + .../2024.11.3/values-ha.yaml | 63 + .../2024.11.3/values.yaml | 671 + .../linkerd-crds/2024.11.3/.helmignore | 22 + .../buoyant/linkerd-crds/2024.11.3/Chart.lock | 6 + .../buoyant/linkerd-crds/2024.11.3/Chart.yaml | 26 + .../buoyant/linkerd-crds/2024.11.3/README.md | 73 + .../linkerd-crds/2024.11.3/README.md.gotmpl | 59 + .../linkerd-crds/2024.11.3/app-readme.md | 9 + .../2024.11.3/charts/partials/.helmignore | 21 + .../2024.11.3/charts/partials/Chart.yaml | 5 + .../2024.11.3/charts/partials/README.md | 9 + .../charts/partials/README.md.gotmpl | 14 + .../charts/partials/templates/NOTES.txt | 0 .../charts/partials/templates/_affinity.tpl | 38 + .../partials/templates/_capabilities.tpl | 16 + .../charts/partials/templates/_debug.tpl | 15 + .../charts/partials/templates/_helpers.tpl | 14 + .../charts/partials/templates/_metadata.tpl | 17 + .../partials/templates/_network-validator.tpl | 45 + .../partials/templates/_nodeselector.tpl | 4 + .../partials/templates/_proxy-config-ann.tpl | 18 + .../charts/partials/templates/_proxy-init.tpl | 98 + .../charts/partials/templates/_proxy.tpl | 271 + .../partials/templates/_pull-secrets.tpl | 6 + .../charts/partials/templates/_resources.tpl | 28 + .../partials/templates/_tolerations.tpl | 4 + .../charts/partials/templates/_trace.tpl | 5 + .../charts/partials/templates/_validate.tpl | 19 + .../charts/partials/templates/_volumes.tpl | 41 + .../2024.11.3/charts/partials/values.yaml | 0 .../2024.11.3/templates/NOTES.txt | 6 + .../gateway.networking.k8s.io_grpcroutes.yaml | 1507 ++ .../gateway.networking.k8s.io_httproutes.yaml | 3881 +++ .../gateway.networking.k8s.io_tcproutes.yaml | 533 + .../gateway.networking.k8s.io_tlsroutes.yaml | 582 + .../policy/authorization-policy.yaml | 99 + .../templates/policy/egress-network.yaml | 123 + .../policy/http-local-ratelimit-policy.yaml | 215 + .../2024.11.3/templates/policy/httproute.yaml | 5328 +++++ .../policy/meshtls-authentication.yaml | 87 + .../policy/network-authentication.yaml | 53 + .../policy/server-authorization.yaml | 266 + .../2024.11.3/templates/policy/server.yaml | 319 + .../2024.11.3/templates/serviceprofile.yaml | 274 + .../templates/workload/external-workload.yaml | 303 + .../linkerd-crds/2024.11.3/values.yaml | 3 + .../instana/instana-agent/2.0.2/.helmignore | 23 + charts/instana/instana-agent/2.0.2/Chart.yaml | 39 + .../instana/instana-agent/2.0.2/DEPLOYMENT.md | 54 + charts/instana/instana-agent/2.0.2/README.md | 701 + .../instana/instana-agent/2.0.2/app-readme.md | 5 + ...omresourcedefinition_agents_instana_io.yml | 4451 ++++ .../2.0.2/kubernetes.deployment.enabled.png | Bin 0 -> 71985 bytes .../kubernetes_deployment_mode_diagram.py | 20 + .../instana/instana-agent/2.0.2/questions.yml | 236 + .../instana-agent/2.0.2/templates/NOTES.txt | 73 + .../instana-agent/2.0.2/templates/agent.yml | 269 + ...rator_clusterrole_leader-election-role.yml | 27 + .../operator_clusterrole_manager-role.yml | 176 + ...olebinding_leader-election-rolebinding.yml | 13 + ...clusterrolebinding_manager-rolebinding.yml | 13 + .../operator_configmap_manager-config.yml | 17 + ...operator_deployment_controller-manager.yml | 66 + ..._serviceaccount_instana-agent-operator.yml | 6 + .../instana/instana-agent/2.0.2/values.yaml | 271 + charts/percona/psmdb-db/1.18.0/.helmignore | 22 + charts/percona/psmdb-db/1.18.0/Chart.yaml | 19 + charts/percona/psmdb-db/1.18.0/README.md | 316 + .../psmdb-db/1.18.0/templates/NOTES.txt | 40 + .../psmdb-db/1.18.0/templates/_helpers.tpl | 45 + .../1.18.0/templates/cluster-secret.yaml | 12 + .../psmdb-db/1.18.0/templates/cluster.yaml | 676 + charts/percona/psmdb-db/1.18.0/values.yaml | 678 + .../percona/psmdb-operator/1.18.0/.helmignore | 22 + .../percona/psmdb-operator/1.18.0/Chart.yaml | 20 + .../percona/psmdb-operator/1.18.0/LICENSE.txt | 13 + .../percona/psmdb-operator/1.18.0/README.md | 69 + .../psmdb-operator/1.18.0/crds/crd.yaml | 19526 ++++++++++++++++ .../psmdb-operator/1.18.0/templates/NOTES.txt | 15 + .../1.18.0/templates/_helpers.tpl | 45 + .../1.18.0/templates/deployment.yaml | 108 + .../1.18.0/templates/namespace.yaml | 11 + .../1.18.0/templates/role-binding.yaml | 41 + .../psmdb-operator/1.18.0/templates/role.yaml | 167 + .../percona/psmdb-operator/1.18.0/values.yaml | 99 + charts/redpanda/redpanda/5.9.10/.helmignore | 28 + charts/redpanda/redpanda/5.9.10/Chart.lock | 9 + charts/redpanda/redpanda/5.9.10/Chart.yaml | 38 + charts/redpanda/redpanda/5.9.10/LICENSE | 201 + charts/redpanda/redpanda/5.9.10/README.md | 1220 + .../5.9.10/charts/connectors/.helmignore | 29 + .../5.9.10/charts/connectors/Chart.yaml | 25 + .../redpanda/5.9.10/charts/connectors/LICENSE | 201 + .../5.9.10/charts/connectors/README.md | 574 + .../connectors/templates/_deployment.go.tpl | 136 + .../connectors/templates/_helpers.go.tpl | 131 + .../charts/connectors/templates/_helpers.tpl | 79 + .../connectors/templates/_pod-monitor.go.tpl | 18 + .../connectors/templates/_service.go.tpl | 20 + .../templates/_serviceaccount.go.tpl | 18 + .../charts/connectors/templates/_shims.tpl | 289 + .../connectors/templates/_values.go.tpl | 15 + .../connectors/templates/deployment.yaml | 17 + .../connectors/templates/pod-monitor.yaml | 17 + .../charts/connectors/templates/service.yaml | 17 + .../connectors/templates/serviceaccount.yaml | 17 + .../templates/tests/01-mm2-values.yaml | 176 + .../5.9.10/charts/connectors/values.yaml | 311 + .../5.9.10/charts/console/.helmignore | 28 + .../redpanda/5.9.10/charts/console/Chart.yaml | 23 + .../redpanda/5.9.10/charts/console/README.md | 353 + .../console/examples/console-enterprise.yaml | 94 + .../5.9.10/charts/console/templates/NOTES.txt | 20 + .../charts/console/templates/_chart.go.tpl | 13 + .../console/templates/_configmap.go.tpl | 25 + .../console/templates/_deployment.go.tpl | 133 + .../charts/console/templates/_helpers.go.tpl | 82 + .../charts/console/templates/_helpers.tpl | 25 + .../charts/console/templates/_hpa.go.tpl | 25 + .../charts/console/templates/_ingress.go.tpl | 46 + .../charts/console/templates/_notes.go.tpl | 40 + .../charts/console/templates/_secret.go.tpl | 22 + .../charts/console/templates/_service.go.tpl | 20 + .../console/templates/_serviceaccount.go.tpl | 39 + .../charts/console/templates/_shims.tpl | 289 + .../charts/console/templates/entry-point.yaml | 17 + .../templates/tests/test-connection.yaml | 22 + .../5.9.10/charts/console/values.schema.json | 323 + .../5.9.10/charts/console/values.yaml | 279 + .../redpanda/5.9.10/templates/NOTES.txt | 26 + .../5.9.10/templates/_cert-issuers.go.tpl | 57 + .../redpanda/5.9.10/templates/_certs.go.tpl | 71 + .../redpanda/5.9.10/templates/_chart.go.tpl | 62 + .../5.9.10/templates/_configmap.go.tpl | 531 + .../redpanda/5.9.10/templates/_console.go.tpl | 178 + .../5.9.10/templates/_example-commands.tpl | 58 + .../redpanda/5.9.10/templates/_helpers.go.tpl | 570 + .../redpanda/5.9.10/templates/_helpers.tpl | 368 + .../redpanda/5.9.10/templates/_memory.go.tpl | 63 + .../redpanda/5.9.10/templates/_notes.go.tpl | 167 + .../templates/_poddisruptionbudget.go.tpl | 21 + .../_post-install-upgrade-job.go.tpl | 123 + .../5.9.10/templates/_post_upgrade_job.go.tpl | 87 + .../redpanda/5.9.10/templates/_rbac.go.tpl | 116 + .../redpanda/5.9.10/templates/_secrets.go.tpl | 419 + .../5.9.10/templates/_service.internal.go.tpl | 38 + .../templates/_service.loadbalancer.go.tpl | 105 + .../5.9.10/templates/_service.nodeport.go.tpl | 80 + .../5.9.10/templates/_serviceaccount.go.tpl | 18 + .../5.9.10/templates/_servicemonitor.go.tpl | 26 + .../redpanda/5.9.10/templates/_shims.tpl | 339 + .../5.9.10/templates/_statefulset.go.tpl | 777 + .../redpanda/5.9.10/templates/_values.go.tpl | 1351 ++ .../templates/connectors/connectors.yaml | 109 + .../5.9.10/templates/entry-point.yaml | 17 + .../templates/tests/test-api-status.yaml | 52 + .../templates/tests/test-auditLogging.yaml | 86 + .../tests/test-connector-via-console.yaml | 166 + .../5.9.10/templates/tests/test-console.yaml | 49 + .../test-internal-external-tls-secrets.yaml | 122 + .../tests/test-kafka-internal-tls-status.yaml | 62 + .../templates/tests/test-kafka-nodelete.yaml | 100 + .../tests/test-kafka-produce-consume.yaml | 83 + .../tests/test-kafka-sasl-status.yaml | 79 + .../tests/test-license-with-console.yaml | 61 + .../tests/test-lifecycle-scripts.yaml | 66 + .../tests/test-loadbalancer-tls.yaml | 173 + .../templates/tests/test-nodeport-tls.yaml | 173 + .../test-pandaproxy-internal-tls-status.yaml | 81 + .../tests/test-pandaproxy-status.yaml | 72 + .../tests/test-prometheus-targets.yaml | 84 + .../templates/tests/test-rack-awareness.yaml | 61 + .../tests/test-rpk-debug-bundle.yaml | 104 + .../templates/tests/test-sasl-updated.yaml | 71 + .../redpanda/5.9.10/values.schema.json | 12882 ++++++++++ charts/redpanda/redpanda/5.9.10/values.yaml | 1133 + index.yaml | 198 +- 228 files changed, 74048 insertions(+), 3 deletions(-) create mode 100644 assets/buoyant/linkerd-control-plane-2024.11.3.tgz create mode 100644 assets/buoyant/linkerd-crds-2024.11.3.tgz create mode 100644 assets/instana/instana-agent-2.0.2.tgz create mode 100644 assets/percona/psmdb-db-1.18.0.tgz create mode 100644 assets/percona/psmdb-operator-1.18.0.tgz create mode 100644 assets/redpanda/redpanda-5.9.10.tgz create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/.helmignore create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/Chart.lock create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/Chart.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/README.md create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/README.md.gotmpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/app-readme.md create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/.helmignore create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/Chart.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md.gotmpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/NOTES.txt create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_affinity.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_capabilities.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_debug.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_helpers.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_metadata.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_network-validator.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_nodeselector.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-init.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_pull-secrets.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_resources.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_tolerations.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_trace.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_validate.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_volumes.tpl create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/values.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/questions.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/NOTES.txt create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/config-rbac.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/config.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination-rbac.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat-rbac.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity-rbac.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/namespace.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/podmonitor.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector-rbac.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/templates/psp.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/values-ha.yaml create mode 100644 charts/buoyant/linkerd-control-plane/2024.11.3/values.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/.helmignore create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/Chart.lock create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/Chart.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/README.md create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/README.md.gotmpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/app-readme.md create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/.helmignore create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/Chart.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md.gotmpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/NOTES.txt create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_affinity.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_capabilities.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_debug.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_helpers.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_metadata.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_network-validator.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_nodeselector.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-init.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_pull-secrets.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_resources.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_tolerations.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_trace.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_validate.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_volumes.tpl create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/charts/partials/values.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/NOTES.txt create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_grpcroutes.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_httproutes.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tcproutes.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tlsroutes.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/authorization-policy.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/egress-network.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/http-local-ratelimit-policy.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/httproute.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/meshtls-authentication.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/network-authentication.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server-authorization.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/serviceprofile.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/templates/workload/external-workload.yaml create mode 100644 charts/buoyant/linkerd-crds/2024.11.3/values.yaml create mode 100644 charts/instana/instana-agent/2.0.2/.helmignore create mode 100644 charts/instana/instana-agent/2.0.2/Chart.yaml create mode 100644 charts/instana/instana-agent/2.0.2/DEPLOYMENT.md create mode 100644 charts/instana/instana-agent/2.0.2/README.md create mode 100644 charts/instana/instana-agent/2.0.2/app-readme.md create mode 100644 charts/instana/instana-agent/2.0.2/crds/operator_customresourcedefinition_agents_instana_io.yml create mode 100644 charts/instana/instana-agent/2.0.2/kubernetes.deployment.enabled.png create mode 100644 charts/instana/instana-agent/2.0.2/kubernetes_deployment_mode_diagram.py create mode 100644 charts/instana/instana-agent/2.0.2/questions.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/NOTES.txt create mode 100644 charts/instana/instana-agent/2.0.2/templates/agent.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_leader-election-role.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_manager-role.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_leader-election-rolebinding.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_manager-rolebinding.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_configmap_manager-config.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_deployment_controller-manager.yml create mode 100644 charts/instana/instana-agent/2.0.2/templates/operator_serviceaccount_instana-agent-operator.yml create mode 100644 charts/instana/instana-agent/2.0.2/values.yaml create mode 100644 charts/percona/psmdb-db/1.18.0/.helmignore create mode 100644 charts/percona/psmdb-db/1.18.0/Chart.yaml create mode 100644 charts/percona/psmdb-db/1.18.0/README.md create mode 100644 charts/percona/psmdb-db/1.18.0/templates/NOTES.txt create mode 100644 charts/percona/psmdb-db/1.18.0/templates/_helpers.tpl create mode 100644 charts/percona/psmdb-db/1.18.0/templates/cluster-secret.yaml create mode 100644 charts/percona/psmdb-db/1.18.0/templates/cluster.yaml create mode 100644 charts/percona/psmdb-db/1.18.0/values.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/.helmignore create mode 100644 charts/percona/psmdb-operator/1.18.0/Chart.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/LICENSE.txt create mode 100644 charts/percona/psmdb-operator/1.18.0/README.md create mode 100644 charts/percona/psmdb-operator/1.18.0/crds/crd.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/NOTES.txt create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/_helpers.tpl create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/deployment.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/namespace.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/role-binding.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/templates/role.yaml create mode 100644 charts/percona/psmdb-operator/1.18.0/values.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/.helmignore create mode 100644 charts/redpanda/redpanda/5.9.10/Chart.lock create mode 100644 charts/redpanda/redpanda/5.9.10/Chart.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/LICENSE create mode 100644 charts/redpanda/redpanda/5.9.10/README.md create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/.helmignore create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/Chart.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/LICENSE create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/README.md create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_deployment.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_pod-monitor.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_service.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_serviceaccount.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_shims.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_values.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/deployment.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/pod-monitor.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/service.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/serviceaccount.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/templates/tests/01-mm2-values.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/connectors/values.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/.helmignore create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/Chart.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/README.md create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/examples/console-enterprise.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/NOTES.txt create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_chart.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_configmap.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_deployment.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_hpa.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_ingress.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_notes.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_secret.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_service.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_serviceaccount.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/_shims.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/entry-point.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/templates/tests/test-connection.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/values.schema.json create mode 100644 charts/redpanda/redpanda/5.9.10/charts/console/values.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/NOTES.txt create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_cert-issuers.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_certs.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_chart.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_configmap.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_console.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_example-commands.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_helpers.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_helpers.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_memory.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_notes.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_poddisruptionbudget.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_post-install-upgrade-job.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_post_upgrade_job.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_rbac.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_secrets.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_service.internal.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_service.loadbalancer.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_service.nodeport.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_serviceaccount.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_servicemonitor.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_shims.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_statefulset.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/_values.go.tpl create mode 100644 charts/redpanda/redpanda/5.9.10/templates/connectors/connectors.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/entry-point.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-api-status.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-auditLogging.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-connector-via-console.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-console.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-internal-external-tls-secrets.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-internal-tls-status.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-nodelete.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-produce-consume.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-sasl-status.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-license-with-console.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-lifecycle-scripts.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-loadbalancer-tls.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-nodeport-tls.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-internal-tls-status.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-status.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-prometheus-targets.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-rack-awareness.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-rpk-debug-bundle.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/templates/tests/test-sasl-updated.yaml create mode 100644 charts/redpanda/redpanda/5.9.10/values.schema.json create mode 100644 charts/redpanda/redpanda/5.9.10/values.yaml diff --git a/assets/buoyant/linkerd-control-plane-2024.11.2.tgz b/assets/buoyant/linkerd-control-plane-2024.11.2.tgz index d507ac6ffdd44d72920f0fbd572ff4752ba3dbe8..509cb7edcdb40006489106173186cb4a19ff0112 100644 GIT binary patch delta 30809 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 delta 30678 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

1_%4P=sw=>J^5F~1idaS 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.{$Wfd_?DH=L-JH=i#xm_!5Zgk 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 literal 0 HcmV?d00001 diff --git a/assets/instana/instana-agent-2.0.2.tgz b/assets/instana/instana-agent-2.0.2.tgz new file mode 100644 index 0000000000000000000000000000000000000000..ff947abc92b5024e929b56eddd13875ee8757129 GIT binary patch literal 116586 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 w)!J_RCu@1h6t?kmy&a6ke74_>4GG5`Po literal 0 HcmV?d00001 diff --git a/assets/percona/psmdb-operator-1.18.0.tgz b/assets/percona/psmdb-operator-1.18.0.tgz new file mode 100644 index 0000000000000000000000000000000000000000..7421ac954ab2c3c2588a64ad173b4c9c65c69ef5 GIT binary patch literal 52509 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�u~ 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±}gn$F2=L=uaXzL12Qf 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.22.0-0' catalog.cattle.io/release-name: linkerd-control-plane apiVersion: v2 diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/.helmignore b/charts/buoyant/linkerd-control-plane/2024.11.3/.helmignore new file mode 100644 index 0000000000..79c90a8063 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +OWNERS +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.lock b/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.lock new file mode 100644 index 0000000000..a0cb7ec8c5 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: partials + repository: file://../partials + version: 0.1.0 +digest: sha256:8e42f9c9d4a2dc883f17f94d6044c97518ced19ad0922f47b8760e47135369ba +generated: "2021-12-06T11:42:50.784240359-05:00" diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.yaml new file mode 100644 index 0000000000..45d987fc69 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/Chart.yaml @@ -0,0 +1,29 @@ +annotations: + catalog.cattle.io/auto-install: linkerd-crds + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Linkerd Control Plane + catalog.cattle.io/featured: "5" + catalog.cattle.io/kube-version: '>=1.22.0-0' + catalog.cattle.io/release-name: linkerd-control-plane +apiVersion: v2 +appVersion: edge-24.11.3 +dependencies: +- name: partials + repository: file://../partials + version: 0.1.0 +description: 'Linkerd gives you observability, reliability, and security for your + microservices — with no code change required. ' +home: https://linkerd.io +icon: file://assets/icons/linkerd-control-plane.png +keywords: +- service-mesh +kubeVersion: '>=1.22.0-0' +maintainers: +- email: cncf-linkerd-dev@lists.cncf.io + name: Linkerd authors + url: https://linkerd.io/ +name: linkerd-control-plane +sources: +- https://github.com/linkerd/linkerd2/ +type: application +version: 2024.11.3 diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/README.md b/charts/buoyant/linkerd-control-plane/2024.11.3/README.md new file mode 100644 index 0000000000..80afb2fac9 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/README.md @@ -0,0 +1,322 @@ +# linkerd-control-plane + +Linkerd gives you observability, reliability, and security +for your microservices — with no code change required. + +![Version: 2024.11.3](https://img.shields.io/badge/Version-2024.11.3-informational?style=flat-square) +![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) +![AppVersion: edge-XX.X.X](https://img.shields.io/badge/AppVersion-edge--XX.X.X-informational?style=flat-square) + +**Homepage:** + +## Quickstart and documentation + +You can run Linkerd on any Kubernetes cluster in a matter of seconds. See the +[Linkerd Getting Started Guide][getting-started] for how. + +For more comprehensive documentation, start with the [Linkerd +docs][linkerd-docs]. + +## Prerequisite: linkerd-crds chart + +Before installing this chart, please install the `linkerd-crds` chart, which +creates all the CRDs that the components from the current chart require. + +## Prerequisite: identity certificates + +The identity component of Linkerd requires setting up a trust anchor +certificate, and an issuer certificate with its key. These need to be provided +to Helm by the user (unlike when using the `linkerd install` CLI which can +generate these automatically). You can provide your own, or follow [these +instructions](https://linkerd.io/2/tasks/generate-certificates/) to generate new +ones. + +Alternatively, both trust anchor and identity issuer certificates may be +derived from in-cluster resources. Existing CA (trust anchor) certificates +**must** live in a `ConfigMap` resource named `linkerd-identity-trust-roots`. +Issuer certificates **must** live in a `Secret` named +`linkerd-identity-issuer`. Both resources should exist in the control-plane's +install namespace. In order to use an existing CA, Linkerd needs to be +installed with `identity.externalCA=true`. To use an existing issuer +certificate, Linkerd should be installed with +`identity.issuer.scheme=kubernetes.io/tls`. + +A more comprehensive description is in the [automatic certificate rotation +guide](https://linkerd.io/2.12/tasks/automatically-rotating-control-plane-tls-credentials/#a-note-on-third-party-cert-management-solutions). + +Note that the provided certificates must be ECDSA certificates. + +## Adding Linkerd's Helm repository + +Included here for completeness-sake, but should have already been added when +`linkerd-base` was installed. + +```bash +# To add the repo for Linkerd edge releases: +helm repo add linkerd https://helm.linkerd.io/edge +``` + +## Installing the chart + +You must provide the certificates and keys described in the preceding section, +and the same expiration date you used to generate the Issuer certificate. + +```bash +helm install linkerd-control-plane -n linkerd \ + --set-file identityTrustAnchorsPEM=ca.crt \ + --set-file identity.issuer.tls.crtPEM=issuer.crt \ + --set-file identity.issuer.tls.keyPEM=issuer.key \ + linkerd/linkerd-control-plane +``` + +Note that you require to install this chart in the same namespace you installed +the `linkerd-base` chart. + +## Setting High-Availability + +Besides the default `values.yaml` file, the chart provides a `values-ha.yaml` +file that overrides some default values as to set things up under a +high-availability scenario, analogous to the `--ha` option in `linkerd install`. +Values such as higher number of replicas, higher memory/cpu limits and +affinities are specified in that file. + +You can get ahold of `values-ha.yaml` by fetching the chart files: + +```bash +helm fetch --untar linkerd/linkerd-control-plane +``` + +Then use the `-f` flag to provide the override file, for example: + +```bash +helm install linkerd-control-plane -n linkerd \ + --set-file identityTrustAnchorsPEM=ca.crt \ + --set-file identity.issuer.tls.crtPEM=issuer.crt \ + --set-file identity.issuer.tls.keyPEM=issuer.key \ + -f linkerd2/values-ha.yaml + linkerd/linkerd-control-plane +``` + +## Get involved + +* Check out Linkerd's source code at [GitHub][linkerd2]. +* Join Linkerd's [user mailing list][linkerd-users], [developer mailing + list][linkerd-dev], and [announcements mailing list][linkerd-announce]. +* Follow [@linkerd][twitter] on Twitter. +* Join the [Linkerd Slack][slack]. + +[getting-started]: https://linkerd.io/2/getting-started/ +[linkerd2]: https://github.com/linkerd/linkerd2 +[linkerd-announce]: https://lists.cncf.io/g/cncf-linkerd-announce +[linkerd-dev]: https://lists.cncf.io/g/cncf-linkerd-dev +[linkerd-docs]: https://linkerd.io/2/overview/ +[linkerd-users]: https://lists.cncf.io/g/cncf-linkerd-users +[slack]: http://slack.linkerd.io +[twitter]: https://twitter.com/linkerd + +## Extensions for Linkerd + +The current chart installs the core Linkerd components, which grant you +reliability and security features. Other functionality is available through +extensions. Check the corresponding docs for each one of the following +extensions: + +* Observability: + [Linkerd-viz](https://github.com/linkerd/linkerd2/blob/main/viz/charts/linkerd-viz/README.md) +* Multicluster: + [Linkerd-multicluster](https://github.com/linkerd/linkerd2/blob/main/multicluster/charts/linkerd-multicluster/README.md) +* Tracing: + [Linkerd-jaeger](https://github.com/linkerd/linkerd2/blob/main/jaeger/charts/linkerd-jaeger/README.md) + +## Requirements + +Kubernetes: `>=1.22.0-0` + +| Repository | Name | Version | +|------------|------|---------| +| file://../partials | partials | 0.1.0 | + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| clusterDomain | string | `"cluster.local"` | Kubernetes DNS Domain name to use | +| clusterNetworks | string | `"10.0.0.0/8,100.64.0.0/10,172.16.0.0/12,192.168.0.0/16,fd00::/8"` | The cluster networks for which service discovery is performed. This should include the pod and service networks, but need not include the node network. By default, all IPv4 private networks and all accepted IPv6 ULAs are specified so that resolution works in typical Kubernetes environments. | +| cniEnabled | bool | `false` | enabling this omits the NET_ADMIN capability in the PSP and the proxy-init container when injecting the proxy; requires the linkerd-cni plugin to already be installed | +| commonLabels | object | `{}` | Labels to apply to all resources | +| controlPlaneTracing | bool | `false` | enables control plane tracing | +| controlPlaneTracingNamespace | string | `"linkerd-jaeger"` | namespace to send control plane traces to | +| controller.podDisruptionBudget | object | `{"maxUnavailable":1}` | sets pod disruption budget parameter for all deployments | +| controller.podDisruptionBudget.maxUnavailable | int | `1` | Maximum number of pods that can be unavailable during disruption | +| controllerGID | int | `-1` | Optional customisation of the group ID for the control plane components (the group ID will be omitted if lower than 0) | +| controllerImage | string | `"cr.l5d.io/linkerd/controller"` | Docker image for the destination and identity components | +| controllerImageVersion | string | `""` | Optionally allow a specific container image Tag (or SHA) to be specified for the controllerImage. | +| controllerLogFormat | string | `"plain"` | Log format for the control plane components | +| controllerLogLevel | string | `"info"` | Log level for the control plane components | +| controllerReplicas | int | `1` | Number of replicas for each control plane pod | +| controllerUID | int | `2103` | User ID for the control plane components | +| debugContainer.image.name | string | `"cr.l5d.io/linkerd/debug"` | Docker image for the debug container | +| debugContainer.image.pullPolicy | string | imagePullPolicy | Pull policy for the debug container image | +| debugContainer.image.version | string | linkerdVersion | Tag for the debug container image | +| deploymentStrategy | object | `{"rollingUpdate":{"maxSurge":"25%","maxUnavailable":"25%"}}` | default kubernetes deployment strategy | +| destinationController.livenessProbe.timeoutSeconds | int | `1` | | +| destinationController.meshedHttp2ClientProtobuf.keep_alive.interval.seconds | int | `10` | | +| destinationController.meshedHttp2ClientProtobuf.keep_alive.timeout.seconds | int | `3` | | +| destinationController.meshedHttp2ClientProtobuf.keep_alive.while_idle | bool | `true` | | +| destinationController.readinessProbe.timeoutSeconds | int | `1` | | +| disableHeartBeat | bool | `false` | Set to true to not start the heartbeat cronjob | +| disableIPv6 | bool | `true` | disables routing IPv6 traffic in addition to IPv4 traffic through the proxy (IPv6 routing only available as of proxy-init v2.3.0 and linkerd-cni v1.4.0) | +| egress.globalEgressNetworkNamespace | string | `"linkerd-egress"` | The namespace that is used to store egress configuration that affects all client workloads in the cluster | +| enableEndpointSlices | bool | `true` | enables the use of EndpointSlice informers for the destination service; enableEndpointSlices should be set to true only if EndpointSlice K8s feature gate is on | +| enableH2Upgrade | bool | `true` | Allow proxies to perform transparent HTTP/2 upgrading | +| enablePSP | bool | `false` | Add a PSP resource and bind it to the control plane ServiceAccounts. Note PSP has been deprecated since k8s v1.21 | +| enablePodAntiAffinity | bool | `false` | enables pod anti affinity creation on deployments for high availability | +| enablePodDisruptionBudget | bool | `false` | enables the creation of pod disruption budgets for control plane components | +| enablePprof | bool | `false` | enables the use of pprof endpoints on control plane component's admin servers | +| identity.externalCA | bool | `false` | If the linkerd-identity-trust-roots ConfigMap has already been created | +| identity.issuer.clockSkewAllowance | string | `"20s"` | Amount of time to allow for clock skew within a Linkerd cluster | +| identity.issuer.issuanceLifetime | string | `"24h0m0s"` | Amount of time for which the Identity issuer should certify identity | +| identity.issuer.scheme | string | `"linkerd.io/tls"` | | +| identity.issuer.tls | object | `{"crtPEM":"","keyPEM":""}` | Which scheme is used for the identity issuer secret format | +| identity.issuer.tls.crtPEM | string | `""` | Issuer certificate (ECDSA). It must be provided during install. | +| identity.issuer.tls.keyPEM | string | `""` | Key for the issuer certificate (ECDSA). It must be provided during install | +| identity.kubeAPI.clientBurst | int | `200` | Burst value over clientQPS | +| identity.kubeAPI.clientQPS | int | `100` | Maximum QPS sent to the kube-apiserver before throttling. See [token bucket rate limiter implementation](https://github.com/kubernetes/client-go/blob/v12.0.0/util/flowcontrol/throttle.go) | +| identity.livenessProbe.timeoutSeconds | int | `1` | | +| identity.readinessProbe.timeoutSeconds | int | `1` | | +| identity.serviceAccountTokenProjection | bool | `true` | Use [Service Account token Volume projection](https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#service-account-token-volume-projection) for pod validation instead of the default token | +| identityTrustAnchorsPEM | string | `""` | Trust root certificate (ECDSA). It must be provided during install. | +| identityTrustDomain | string | clusterDomain | Trust domain used for identity | +| imagePullPolicy | string | `"IfNotPresent"` | Docker image pull policy | +| imagePullSecrets | list | `[]` | For Private docker registries, authentication is needed. Registry secrets are applied to the respective service accounts | +| kubeAPI.clientBurst | int | `200` | Burst value over clientQPS | +| kubeAPI.clientQPS | int | `100` | Maximum QPS sent to the kube-apiserver before throttling. See [token bucket rate limiter implementation](https://github.com/kubernetes/client-go/blob/v12.0.0/util/flowcontrol/throttle.go) | +| linkerdVersion | string | `"linkerdVersionValue"` | control plane version. See Proxy section for proxy version | +| networkValidator.connectAddr | string | `""` | Address to which the network-validator will attempt to connect. This should be an IP that the cluster is expected to be able to reach but a port it should not, e.g., a public IP for public clusters and a private IP for air-gapped clusters with a port like 20001. If empty, defaults to 1.1.1.1:20001 and [fd00::1]:20001 for IPv4 and IPv6 respectively. | +| networkValidator.enableSecurityContext | bool | `true` | Include a securityContext in the network-validator pod spec | +| networkValidator.listenAddr | string | `""` | Address to which network-validator listens to requests from itself. If empty, defaults to 0.0.0.0:4140 and [::]:4140 for IPv4 and IPv6 respectively. | +| networkValidator.logFormat | string | plain | Log format (`plain` or `json`) for network-validator | +| networkValidator.logLevel | string | debug | Log level for the network-validator | +| networkValidator.timeout | string | `"10s"` | Timeout before network-validator fails to validate the pod's network connectivity | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector section, See the [K8S documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector) for more information | +| podAnnotations | object | `{}` | Additional annotations to add to all pods | +| podLabels | object | `{}` | Additional labels to add to all pods | +| podMonitor.controller.enabled | bool | `true` | Enables the creation of PodMonitor for the control-plane | +| podMonitor.controller.namespaceSelector | string | `"matchNames:\n - {{ .Release.Namespace }}\n - linkerd-viz\n - linkerd-jaeger\n"` | Selector to select which namespaces the Endpoints objects are discovered from | +| podMonitor.enabled | bool | `false` | Enables the creation of Prometheus Operator [PodMonitor](https://prometheus-operator.dev/docs/operator/api/#monitoring.coreos.com/v1.PodMonitor) | +| podMonitor.labels | object | `{}` | Labels to apply to all pod Monitors | +| podMonitor.proxy.enabled | bool | `true` | Enables the creation of PodMonitor for the data-plane | +| podMonitor.scrapeInterval | string | `"10s"` | Interval at which metrics should be scraped | +| podMonitor.scrapeTimeout | string | `"10s"` | Iimeout after which the scrape is ended | +| podMonitor.serviceMirror.enabled | bool | `true` | Enables the creation of PodMonitor for the Service Mirror component | +| policyController.image.name | string | `"cr.l5d.io/linkerd/policy-controller"` | Docker image for the policy controller | +| policyController.image.pullPolicy | string | imagePullPolicy | Pull policy for the policy controller container image | +| policyController.image.version | string | linkerdVersion | Tag for the policy controller container image | +| policyController.livenessProbe.timeoutSeconds | int | `1` | | +| policyController.logLevel | string | `"info"` | Log level for the policy controller | +| policyController.probeNetworks | list | `["0.0.0.0/0","::/0"]` | The networks from which probes are performed. By default, all networks are allowed so that all probes are authorized. | +| policyController.readinessProbe.timeoutSeconds | int | `1` | | +| policyController.resources | object | `{"cpu":{"limit":"","request":""},"ephemeral-storage":{"limit":"","request":""},"memory":{"limit":"","request":""}}` | policy controller resource requests & limits | +| policyController.resources.cpu.limit | string | `""` | Maximum amount of CPU units that the policy controller can use | +| policyController.resources.cpu.request | string | `""` | Amount of CPU units that the policy controller requests | +| policyController.resources.ephemeral-storage.limit | string | `""` | Maximum amount of ephemeral storage that the policy controller can use | +| policyController.resources.ephemeral-storage.request | string | `""` | Amount of ephemeral storage that the policy controller requests | +| policyController.resources.memory.limit | string | `""` | Maximum amount of memory that the policy controller can use | +| policyController.resources.memory.request | string | `""` | Maximum amount of memory that the policy controller requests | +| policyValidator.caBundle | string | `""` | Bundle of CA certificates for proxy injector. If not provided nor injected with cert-manager, then Helm will use the certificate generated for `policyValidator.crtPEM`. If `policyValidator.externalSecret` is set to true, this value, injectCaFrom, or injectCaFromSecret must be set, as no certificate will be generated. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. | +| policyValidator.crtPEM | string | `""` | Certificate for the policy validator. If not provided and not using an external secret then Helm will generate one. | +| policyValidator.externalSecret | bool | `false` | Do not create a secret resource for the policyValidator webhook. If this is set to `true`, the value `policyValidator.caBundle` must be set or the ca bundle must injected with cert-manager ca injector using `policyValidator.injectCaFrom` or `policyValidator.injectCaFromSecret` (see below). | +| policyValidator.injectCaFrom | string | `""` | Inject the CA bundle from a cert-manager Certificate. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) for more information. | +| policyValidator.injectCaFromSecret | string | `""` | Inject the CA bundle from a Secret. If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. The Secret must have the CA Bundle stored in the `ca.crt` key and have the `cert-manager.io/allow-direct-injection` annotation set to `true`. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) for more information. | +| policyValidator.keyPEM | string | `""` | Certificate key for the policy validator. If not provided and not using an external secret then Helm will generate one. | +| policyValidator.namespaceSelector | object | `{"matchExpressions":[{"key":"config.linkerd.io/admission-webhooks","operator":"NotIn","values":["disabled"]}]}` | Namespace selector used by admission webhook | +| priorityClassName | string | `""` | Kubernetes priorityClassName for the Linkerd Pods | +| profileValidator.caBundle | string | `""` | Bundle of CA certificates for proxy injector. If not provided nor injected with cert-manager, then Helm will use the certificate generated for `profileValidator.crtPEM`. If `profileValidator.externalSecret` is set to true, this value, injectCaFrom, or injectCaFromSecret must be set, as no certificate will be generated. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. | +| profileValidator.crtPEM | string | `""` | Certificate for the service profile validator. If not provided and not using an external secret then Helm will generate one. | +| profileValidator.externalSecret | bool | `false` | Do not create a secret resource for the profileValidator webhook. If this is set to `true`, the value `proxyInjector.caBundle` must be set or the ca bundle must injected with cert-manager ca injector using `proxyInjector.injectCaFrom` or `proxyInjector.injectCaFromSecret` (see below). | +| profileValidator.injectCaFrom | string | `""` | Inject the CA bundle from a cert-manager Certificate. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) for more information. | +| profileValidator.injectCaFromSecret | string | `""` | Inject the CA bundle from a Secret. If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. The Secret must have the CA Bundle stored in the `ca.crt` key and have the `cert-manager.io/allow-direct-injection` annotation set to `true`. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) for more information. | +| profileValidator.keyPEM | string | `""` | Certificate key for the service profile validator. If not provided and not using an external secret then Helm will generate one. | +| profileValidator.namespaceSelector | object | `{"matchExpressions":[{"key":"config.linkerd.io/admission-webhooks","operator":"NotIn","values":["disabled"]}]}` | Namespace selector used by admission webhook | +| prometheusUrl | string | `""` | url of external prometheus instance (used for the heartbeat) | +| proxy.await | bool | `true` | If set, the application container will not start until the proxy is ready | +| proxy.control.streams.idleTimeout | string | `"5m"` | The timeout between consecutive updates from the control plane. | +| proxy.control.streams.initialTimeout | string | `"3s"` | The timeout for the first update from the control plane. | +| proxy.control.streams.lifetime | string | `"1h"` | The maximum duration for a response stream (i.e. before it will be reinitialized). | +| proxy.cores | int | `0` | The `cpu.limit` and `cores` should be kept in sync. The value of `cores` must be an integer and should typically be set by rounding up from the limit. E.g. if cpu.limit is '1500m', cores should be 2. | +| proxy.defaultInboundPolicy | string | "all-unauthenticated" | The default allow policy to use when no `Server` selects a pod. One of: "all-authenticated", "all-unauthenticated", "cluster-authenticated", "cluster-unauthenticated", "deny", "audit" | +| proxy.disableInboundProtocolDetectTimeout | bool | `false` | When set to true, disables the protocol detection timeout on the inbound side of the proxy by setting it to a very high value | +| proxy.disableOutboundProtocolDetectTimeout | bool | `false` | When set to true, disables the protocol detection timeout on the outbound side of the proxy by setting it to a very high value | +| proxy.enableExternalProfiles | bool | `false` | Enable service profiles for non-Kubernetes services | +| proxy.enableShutdownEndpoint | bool | `false` | Enables the proxy's /shutdown admin endpoint | +| proxy.gid | int | `-1` | Optional customisation of the group id under which the proxy runs (the group ID will be omitted if lower than 0) | +| proxy.image.name | string | `"cr.l5d.io/linkerd/proxy"` | Docker image for the proxy | +| proxy.image.pullPolicy | string | imagePullPolicy | Pull policy for the proxy container image | +| proxy.image.version | string | linkerdVersion | Tag for the proxy container image | +| proxy.inbound.server.http2.keepAliveInterval | string | `"10s"` | The interval at which PINGs are issued to remote HTTP/2 clients. | +| proxy.inbound.server.http2.keepAliveTimeout | string | `"3s"` | The timeout within which keep-alive PINGs must be acknowledged on inbound HTTP/2 connections. | +| proxy.inboundConnectTimeout | string | `"100ms"` | Maximum time allowed for the proxy to establish an inbound TCP connection | +| proxy.inboundDiscoveryCacheUnusedTimeout | string | `"90s"` | Maximum time allowed before an unused inbound discovery result is evicted from the cache | +| proxy.livenessProbe | object | `{"initialDelaySeconds":10,"timeoutSeconds":1}` | LivenessProbe timeout and delay configuration | +| proxy.logFormat | string | `"plain"` | Log format (`plain` or `json`) for the proxy | +| proxy.logHTTPHeaders | `off` or `insecure` | `"off"` | If set to `off`, will prevent the proxy from logging HTTP headers. If set to `insecure`, HTTP headers may be logged verbatim. Note that setting this to `insecure` is not alone sufficient to log HTTP headers; the proxy logLevel must also be set to debug. | +| proxy.logLevel | string | `"warn,linkerd=info,hickory=error"` | Log level for the proxy | +| proxy.nativeSidecar | bool | `false` | Enable KEP-753 native sidecars This is an experimental feature. It requires Kubernetes >= 1.29. If enabled, .proxy.waitBeforeExitSeconds should not be used. | +| proxy.opaquePorts | string | `"25,587,3306,4444,5432,6379,9300,11211"` | Default set of opaque ports - SMTP (25,587) server-first - MYSQL (3306) server-first - Galera (4444) server-first - PostgreSQL (5432) server-first - Redis (6379) server-first - ElasticSearch (9300) server-first - Memcached (11211) clients do not issue any preamble, which breaks detection | +| proxy.outbound.server.http2.keepAliveInterval | string | `"10s"` | The interval at which PINGs are issued to local application HTTP/2 clients. | +| proxy.outbound.server.http2.keepAliveTimeout | string | `"3s"` | The timeout within which keep-alive PINGs must be acknowledged on outbound HTTP/2 connections. | +| proxy.outboundConnectTimeout | string | `"1000ms"` | Maximum time allowed for the proxy to establish an outbound TCP connection | +| proxy.outboundDiscoveryCacheUnusedTimeout | string | `"5s"` | Maximum time allowed before an unused outbound discovery result is evicted from the cache | +| proxy.ports.admin | int | `4191` | Admin port for the proxy container | +| proxy.ports.control | int | `4190` | Control port for the proxy container | +| proxy.ports.inbound | int | `4143` | Inbound port for the proxy container | +| proxy.ports.outbound | int | `4140` | Outbound port for the proxy container | +| proxy.readinessProbe | object | `{"initialDelaySeconds":2,"timeoutSeconds":1}` | ReadinessProbe timeout and delay configuration | +| proxy.requireIdentityOnInboundPorts | string | `""` | | +| proxy.resources.cpu.limit | string | `""` | Maximum amount of CPU units that the proxy can use | +| proxy.resources.cpu.request | string | `""` | Amount of CPU units that the proxy requests | +| proxy.resources.ephemeral-storage.limit | string | `""` | Maximum amount of ephemeral storage that the proxy can use | +| proxy.resources.ephemeral-storage.request | string | `""` | Amount of ephemeral storage that the proxy requests | +| proxy.resources.memory.limit | string | `""` | Maximum amount of memory that the proxy can use | +| proxy.resources.memory.request | string | `""` | Maximum amount of memory that the proxy requests | +| proxy.shutdownGracePeriod | string | `""` | Grace period for graceful proxy shutdowns. If this timeout elapses before all open connections have completed, the proxy will terminate forcefully, closing any remaining connections. | +| proxy.startupProbe.failureThreshold | int | `120` | | +| proxy.startupProbe.initialDelaySeconds | int | `0` | | +| proxy.startupProbe.periodSeconds | int | `1` | | +| proxy.uid | int | `2102` | User id under which the proxy runs | +| proxy.waitBeforeExitSeconds | int | `0` | If set the injected proxy sidecars in the data plane will stay alive for at least the given period before receiving the SIGTERM signal from Kubernetes but no longer than the pod's `terminationGracePeriodSeconds`. See [Lifecycle hooks](https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks) for more info on container lifecycle hooks. | +| proxyInit.closeWaitTimeoutSecs | int | `0` | | +| proxyInit.ignoreInboundPorts | string | `"4567,4568"` | Default set of inbound ports to skip via iptables - Galera (4567,4568) | +| proxyInit.ignoreOutboundPorts | string | `"4567,4568"` | Default set of outbound ports to skip via iptables - Galera (4567,4568) | +| proxyInit.image.name | string | `"cr.l5d.io/linkerd/proxy-init"` | Docker image for the proxy-init container | +| proxyInit.image.pullPolicy | string | imagePullPolicy | Pull policy for the proxy-init container image | +| proxyInit.image.version | string | `"v2.4.1"` | Tag for the proxy-init container image | +| proxyInit.iptablesMode | string | `"legacy"` | Variant of iptables that will be used to configure routing. Currently, proxy-init can be run either in 'nft' or in 'legacy' mode. The mode will control which utility binary will be called. The host must support whichever mode will be used | +| proxyInit.kubeAPIServerPorts | string | `"443,6443"` | Default set of ports to skip via iptables for control plane components so they can communicate with the Kubernetes API Server | +| proxyInit.logFormat | string | plain | Log format (`plain` or `json`) for the proxy-init | +| proxyInit.logLevel | string | info | Log level for the proxy-init | +| proxyInit.privileged | bool | false | Privileged mode allows the container processes to inherit all security capabilities and bypass any security limitations enforced by the kubelet. When used with 'runAsRoot: true', the container will behave exactly as if it was running as root on the host. May escape cgroup limits and see other processes and devices on the host. | +| proxyInit.runAsGroup | int | `65534` | This value is used only if runAsRoot is false; otherwise runAsGroup will be 0 | +| proxyInit.runAsRoot | bool | `false` | Allow overriding the runAsNonRoot behaviour () | +| proxyInit.runAsUser | int | `65534` | This value is used only if runAsRoot is false; otherwise runAsUser will be 0 | +| proxyInit.skipSubnets | string | `""` | Comma-separated list of subnets in valid CIDR format that should be skipped by the proxy | +| proxyInit.xtMountPath.mountPath | string | `"/run"` | | +| proxyInit.xtMountPath.name | string | `"linkerd-proxy-init-xtables-lock"` | | +| proxyInjector.caBundle | string | `""` | Bundle of CA certificates for proxy injector. If not provided nor injected with cert-manager, then Helm will use the certificate generated for `proxyInjector.crtPEM`. If `proxyInjector.externalSecret` is set to true, this value, injectCaFrom, or injectCaFromSecret must be set, as no certificate will be generated. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. | +| proxyInjector.crtPEM | string | `""` | Certificate for the proxy injector. If not provided and not using an external secret then Helm will generate one. | +| proxyInjector.externalSecret | bool | `false` | Do not create a secret resource for the proxyInjector webhook. If this is set to `true`, the value `proxyInjector.caBundle` must be set or the ca bundle must injected with cert-manager ca injector using `proxyInjector.injectCaFrom` or `proxyInjector.injectCaFromSecret` (see below). | +| proxyInjector.injectCaFrom | string | `""` | Inject the CA bundle from a cert-manager Certificate. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) for more information. | +| proxyInjector.injectCaFromSecret | string | `""` | Inject the CA bundle from a Secret. If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. The Secret must have the CA Bundle stored in the `ca.crt` key and have the `cert-manager.io/allow-direct-injection` annotation set to `true`. See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) for more information. | +| proxyInjector.keyPEM | string | `""` | Certificate key for the proxy injector. If not provided and not using an external secret then Helm will generate one. | +| proxyInjector.livenessProbe.timeoutSeconds | int | `1` | | +| proxyInjector.namespaceSelector | object | `{"matchExpressions":[{"key":"config.linkerd.io/admission-webhooks","operator":"NotIn","values":["disabled"]},{"key":"kubernetes.io/metadata.name","operator":"NotIn","values":["kube-system","cert-manager"]}]}` | Namespace selector used by admission webhook. | +| proxyInjector.objectSelector | object | `{"matchExpressions":[{"key":"linkerd.io/control-plane-component","operator":"DoesNotExist"},{"key":"linkerd.io/cni-resource","operator":"DoesNotExist"}]}` | Object selector used by admission webhook. | +| proxyInjector.readinessProbe.timeoutSeconds | int | `1` | | +| proxyInjector.timeoutSeconds | int | `10` | Timeout in seconds before the API Server cancels a request to the proxy injector. If timeout is exceeded, the webhookfailurePolicy is used. | +| revisionHistoryLimit | int | `10` | Specifies the number of old ReplicaSets to retain to allow rollback. | +| runtimeClassName | string | `""` | Runtime Class Name for all the pods | +| spValidator | object | `{"livenessProbe":{"timeoutSeconds":1},"readinessProbe":{"timeoutSeconds":1}}` | SP validator configuration | +| webhookFailurePolicy | string | `"Ignore"` | Failure policy for the proxy injector | + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.12.0](https://github.com/norwoodj/helm-docs/releases/v1.12.0) diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/README.md.gotmpl b/charts/buoyant/linkerd-control-plane/2024.11.3/README.md.gotmpl new file mode 100644 index 0000000000..19da2a82d6 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/README.md.gotmpl @@ -0,0 +1,133 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +{{ template "chart.versionBadge" . }} +{{ template "chart.typeBadge" . }} +{{ template "chart.appVersionBadge" . }} + +{{ template "chart.homepageLine" . }} + +## Quickstart and documentation + +You can run Linkerd on any Kubernetes cluster in a matter of seconds. See the +[Linkerd Getting Started Guide][getting-started] for how. + +For more comprehensive documentation, start with the [Linkerd +docs][linkerd-docs]. + +## Prerequisite: linkerd-crds chart + +Before installing this chart, please install the `linkerd-crds` chart, which +creates all the CRDs that the components from the current chart require. + +## Prerequisite: identity certificates + +The identity component of Linkerd requires setting up a trust anchor +certificate, and an issuer certificate with its key. These need to be provided +to Helm by the user (unlike when using the `linkerd install` CLI which can +generate these automatically). You can provide your own, or follow [these +instructions](https://linkerd.io/2/tasks/generate-certificates/) to generate new +ones. + +Alternatively, both trust anchor and identity issuer certificates may be +derived from in-cluster resources. Existing CA (trust anchor) certificates +**must** live in a `ConfigMap` resource named `linkerd-identity-trust-roots`. +Issuer certificates **must** live in a `Secret` named +`linkerd-identity-issuer`. Both resources should exist in the control-plane's +install namespace. In order to use an existing CA, Linkerd needs to be +installed with `identity.externalCA=true`. To use an existing issuer +certificate, Linkerd should be installed with +`identity.issuer.scheme=kubernetes.io/tls`. + +A more comprehensive description is in the [automatic certificate rotation +guide](https://linkerd.io/2.12/tasks/automatically-rotating-control-plane-tls-credentials/#a-note-on-third-party-cert-management-solutions). + +Note that the provided certificates must be ECDSA certificates. + +## Adding Linkerd's Helm repository + +Included here for completeness-sake, but should have already been added when +`linkerd-base` was installed. + +```bash +# To add the repo for Linkerd edge releases: +helm repo add linkerd https://helm.linkerd.io/edge +``` + +## Installing the chart + +You must provide the certificates and keys described in the preceding section, +and the same expiration date you used to generate the Issuer certificate. + +```bash +helm install linkerd-control-plane -n linkerd \ + --set-file identityTrustAnchorsPEM=ca.crt \ + --set-file identity.issuer.tls.crtPEM=issuer.crt \ + --set-file identity.issuer.tls.keyPEM=issuer.key \ + linkerd/linkerd-control-plane +``` + +Note that you require to install this chart in the same namespace you installed +the `linkerd-base` chart. + +## Setting High-Availability + +Besides the default `values.yaml` file, the chart provides a `values-ha.yaml` +file that overrides some default values as to set things up under a +high-availability scenario, analogous to the `--ha` option in `linkerd install`. +Values such as higher number of replicas, higher memory/cpu limits and +affinities are specified in that file. + +You can get ahold of `values-ha.yaml` by fetching the chart files: + +```bash +helm fetch --untar linkerd/linkerd-control-plane +``` + +Then use the `-f` flag to provide the override file, for example: + +```bash +helm install linkerd-control-plane -n linkerd \ + --set-file identityTrustAnchorsPEM=ca.crt \ + --set-file identity.issuer.tls.crtPEM=issuer.crt \ + --set-file identity.issuer.tls.keyPEM=issuer.key \ + -f linkerd2/values-ha.yaml + linkerd/linkerd-control-plane +``` + +## Get involved + +* Check out Linkerd's source code at [GitHub][linkerd2]. +* Join Linkerd's [user mailing list][linkerd-users], [developer mailing + list][linkerd-dev], and [announcements mailing list][linkerd-announce]. +* Follow [@linkerd][twitter] on Twitter. +* Join the [Linkerd Slack][slack]. + +[getting-started]: https://linkerd.io/2/getting-started/ +[linkerd2]: https://github.com/linkerd/linkerd2 +[linkerd-announce]: https://lists.cncf.io/g/cncf-linkerd-announce +[linkerd-dev]: https://lists.cncf.io/g/cncf-linkerd-dev +[linkerd-docs]: https://linkerd.io/2/overview/ +[linkerd-users]: https://lists.cncf.io/g/cncf-linkerd-users +[slack]: http://slack.linkerd.io +[twitter]: https://twitter.com/linkerd + +## Extensions for Linkerd + +The current chart installs the core Linkerd components, which grant you +reliability and security features. Other functionality is available through +extensions. Check the corresponding docs for each one of the following +extensions: + +* Observability: + [Linkerd-viz](https://github.com/linkerd/linkerd2/blob/main/viz/charts/linkerd-viz/README.md) +* Multicluster: + [Linkerd-multicluster](https://github.com/linkerd/linkerd2/blob/main/multicluster/charts/linkerd-multicluster/README.md) +* Tracing: + [Linkerd-jaeger](https://github.com/linkerd/linkerd2/blob/main/jaeger/charts/linkerd-jaeger/README.md) + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +{{ template "helm-docs.versionFooter" . }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/app-readme.md b/charts/buoyant/linkerd-control-plane/2024.11.3/app-readme.md new file mode 100644 index 0000000000..351eac5f0d --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/app-readme.md @@ -0,0 +1,14 @@ +# Linkerd 2 Chart + +Linkerd is an ultra light, ultra simple, ultra powerful service mesh. Linkerd +adds security, observability, and reliability to Kubernetes, without the +complexity. + +This particular Helm chart only installs the control plane core. You will also need to install the +linkerd-crds chart. This chart should be automatically installed along with any other dependencies. +If it is not installed as a dependency, install it first. + +To gain access to the observability features, please install the linkerd-viz chart. +Other extensions are available (multicluster, jaeger) under the linkerd Helm repo. + +Full documentation available at: https://linkerd.io/2/overview/ diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/.helmignore b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/.helmignore @@ -0,0 +1,21 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/Chart.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/Chart.yaml new file mode 100644 index 0000000000..23cfc167e3 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/Chart.yaml @@ -0,0 +1,5 @@ +apiVersion: v1 +description: 'A Helm chart containing Linkerd partial templates, depended by the ''linkerd'' + and ''patch'' charts. ' +name: partials +version: 0.1.0 diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md new file mode 100644 index 0000000000..10805c9b94 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md @@ -0,0 +1,9 @@ +# partials + +A Helm chart containing Linkerd partial templates, +depended by the 'linkerd' and 'patch' charts. + +![Version: 0.1.0](https://img.shields.io/badge/Version-0.1.0-informational?style=flat-square) + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.12.0](https://github.com/norwoodj/helm-docs/releases/v1.12.0) diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md.gotmpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md.gotmpl new file mode 100644 index 0000000000..37f5101061 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/README.md.gotmpl @@ -0,0 +1,14 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +{{ template "chart.versionBadge" . }} +{{ template "chart.typeBadge" . }} +{{ template "chart.appVersionBadge" . }} + +{{ template "chart.homepageLine" . }} + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +{{ template "helm-docs.versionFooter" . }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/NOTES.txt b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/NOTES.txt new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_affinity.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_affinity.tpl new file mode 100644 index 0000000000..5dde1da473 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_affinity.tpl @@ -0,0 +1,38 @@ +{{ define "linkerd.pod-affinity" -}} +podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchExpressions: + - key: {{ default "linkerd.io/control-plane-component" .label }} + operator: In + values: + - {{ .component }} + topologyKey: topology.kubernetes.io/zone + weight: 100 + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: {{ default "linkerd.io/control-plane-component" .label }} + operator: In + values: + - {{ .component }} + topologyKey: kubernetes.io/hostname +{{- end }} + +{{ define "linkerd.node-affinity" -}} +nodeAffinity: +{{- toYaml .Values.nodeAffinity | trim | nindent 2 }} +{{- end }} + +{{ define "linkerd.affinity" -}} +{{- if or .Values.enablePodAntiAffinity .Values.nodeAffinity -}} +affinity: +{{- end }} +{{- if .Values.enablePodAntiAffinity -}} +{{- include "linkerd.pod-affinity" . | nindent 2 }} +{{- end }} +{{- if .Values.nodeAffinity -}} +{{- include "linkerd.node-affinity" . | nindent 2 }} +{{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_capabilities.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_capabilities.tpl new file mode 100644 index 0000000000..a595d74c1f --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_capabilities.tpl @@ -0,0 +1,16 @@ +{{- define "partials.proxy.capabilities" -}} +capabilities: + {{- if .Values.proxy.capabilities.add }} + add: + {{- toYaml .Values.proxy.capabilities.add | trim | nindent 4 }} + {{- end }} + {{- if .Values.proxy.capabilities.drop }} + drop: + {{- toYaml .Values.proxy.capabilities.drop | trim | nindent 4 }} + {{- end }} +{{- end -}} + +{{- define "partials.proxy-init.capabilities.drop" -}} +drop: +{{ toYaml .Values.proxyInit.capabilities.drop | trim }} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_debug.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_debug.tpl new file mode 100644 index 0000000000..4df8cc77bc --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_debug.tpl @@ -0,0 +1,15 @@ +{{- define "partials.debug" -}} +image: {{.Values.debugContainer.image.name}}:{{.Values.debugContainer.image.version | default .Values.linkerdVersion}} +imagePullPolicy: {{.Values.debugContainer.image.pullPolicy | default .Values.imagePullPolicy}} +name: linkerd-debug +terminationMessagePolicy: FallbackToLogsOnError +# some environments require probes, so we provide some infallible ones +livenessProbe: + exec: + command: + - "true" +readinessProbe: + exec: + command: + - "true" +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_helpers.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_helpers.tpl new file mode 100644 index 0000000000..b6cdc34d08 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_helpers.tpl @@ -0,0 +1,14 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Splits a coma separated list into a list of string values. +For example "11,22,55,44" will become "11","22","55","44" +*/}} +{{- define "partials.splitStringList" -}} +{{- if gt (len (toString .)) 0 -}} +{{- $ports := toString . | splitList "," -}} +{{- $last := sub (len $ports) 1 -}} +{{- range $i,$port := $ports -}} +"{{$port}}"{{ternary "," "" (ne $i $last)}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_metadata.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_metadata.tpl new file mode 100644 index 0000000000..04d2f1beab --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_metadata.tpl @@ -0,0 +1,17 @@ +{{- define "partials.annotations.created-by" -}} +linkerd.io/created-by: {{ .Values.cliVersion | default (printf "linkerd/helm %s" ( (.Values.image).version | default .Values.linkerdVersion)) }} +{{- end -}} + +{{- define "partials.proxy.annotations" -}} +linkerd.io/proxy-version: {{.Values.proxy.image.version | default .Values.linkerdVersion}} +cluster-autoscaler.kubernetes.io/safe-to-evict: "true" +linkerd.io/trust-root-sha256: {{ .Values.identityTrustAnchorsPEM | sha256sum }} +{{- end -}} + +{{/* +To add labels to the control-plane components, instead update at individual component manifests as +adding here would also update `spec.selector.matchLabels` which are immutable and would fail upgrades. +*/}} +{{- define "partials.proxy.labels" -}} +linkerd.io/proxy-{{.workloadKind}}: {{.component}} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_network-validator.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_network-validator.tpl new file mode 100644 index 0000000000..276056395f --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_network-validator.tpl @@ -0,0 +1,45 @@ +{{- define "partials.network-validator" -}} +name: linkerd-network-validator +image: {{.Values.proxy.image.name}}:{{.Values.proxy.image.version | default .Values.linkerdVersion }} +imagePullPolicy: {{.Values.proxy.image.pullPolicy | default .Values.imagePullPolicy}} +{{ include "partials.resources" .Values.proxy.resources }} +{{- if or .Values.networkValidator.enableSecurityContext }} +securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsGroup: 65534 + runAsNonRoot: true + runAsUser: 65534 + seccompProfile: + type: RuntimeDefault +{{- end }} +command: + - /usr/lib/linkerd/linkerd2-network-validator +args: + - --log-format + - {{ .Values.networkValidator.logFormat }} + - --log-level + - {{ .Values.networkValidator.logLevel }} + - --connect-addr + {{- if .Values.networkValidator.connectAddr }} + - {{ .Values.networkValidator.connectAddr | quote }} + {{- else if .Values.disableIPv6}} + - "1.1.1.1:20001" + {{- else }} + - "[fd00::1]:20001" + {{- end }} + - --listen-addr + {{- if .Values.networkValidator.listenAddr }} + - {{ .Values.networkValidator.listenAddr | quote }} + {{- else if .Values.disableIPv6}} + - "0.0.0.0:4140" + {{- else }} + - "[::]:4140" + {{- end }} + - --timeout + - {{ .Values.networkValidator.timeout }} + +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_nodeselector.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_nodeselector.tpl new file mode 100644 index 0000000000..4cde0ab16e --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_nodeselector.tpl @@ -0,0 +1,4 @@ +{{- define "linkerd.node-selector" -}} +nodeSelector: +{{- toYaml .Values.nodeSelector | trim | nindent 2 }} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl new file mode 100644 index 0000000000..9651b3bd1a --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl @@ -0,0 +1,18 @@ +{{- define "partials.proxy.config.annotations" -}} +{{- with .cpu }} +{{- with .request -}} +config.linkerd.io/proxy-cpu-request: {{. | quote}} +{{end}} +{{- with .limit -}} +config.linkerd.io/proxy-cpu-limit: {{. | quote}} +{{- end}} +{{- end}} +{{- with .memory }} +{{- with .request }} +config.linkerd.io/proxy-memory-request: {{. | quote}} +{{end}} +{{- with .limit -}} +config.linkerd.io/proxy-memory-limit: {{. | quote}} +{{- end}} +{{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-init.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-init.tpl new file mode 100644 index 0000000000..a307b14073 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy-init.tpl @@ -0,0 +1,98 @@ +{{- define "partials.proxy-init" -}} +args: +{{- if (.Values.proxyInit.iptablesMode | default "legacy" | eq "nft") }} +- --firewall-bin-path +- "iptables-nft" +- --firewall-save-bin-path +- "iptables-nft-save" +{{- else if not (eq .Values.proxyInit.iptablesMode "legacy") }} +{{ fail (printf "Unsupported value \"%s\" for proxyInit.iptablesMode\nValid values: [\"nft\", \"legacy\"]" .Values.proxyInit.iptablesMode) }} +{{end -}} +{{- if .Values.disableIPv6 }} +- --ipv6=false +{{- end }} +- --incoming-proxy-port +- {{.Values.proxy.ports.inbound | quote}} +- --outgoing-proxy-port +- {{.Values.proxy.ports.outbound | quote}} +- --proxy-uid +- {{.Values.proxy.uid | quote}} +{{- if ge (int .Values.proxy.gid) 0 }} +- --proxy-gid +- {{.Values.proxy.gid | quote}} +{{- end }} +- --inbound-ports-to-ignore +- "{{.Values.proxy.ports.control}},{{.Values.proxy.ports.admin}}{{ternary (printf ",%s" (.Values.proxyInit.ignoreInboundPorts | toString)) "" (not (empty .Values.proxyInit.ignoreInboundPorts)) }}" +{{- if .Values.proxyInit.ignoreOutboundPorts }} +- --outbound-ports-to-ignore +- {{.Values.proxyInit.ignoreOutboundPorts | quote}} +{{- end }} +{{- if .Values.proxyInit.closeWaitTimeoutSecs }} +- --timeout-close-wait-secs +- {{ .Values.proxyInit.closeWaitTimeoutSecs | quote}} +{{- end }} +{{- if .Values.proxyInit.logFormat }} +- --log-format +- {{ .Values.proxyInit.logFormat }} +{{- end }} +{{- if .Values.proxyInit.logLevel }} +- --log-level +- {{ .Values.proxyInit.logLevel }} +{{- end }} +{{- if .Values.proxyInit.skipSubnets }} +- --subnets-to-ignore +- {{ .Values.proxyInit.skipSubnets | quote }} +{{- end }} +image: {{.Values.proxyInit.image.name}}:{{.Values.proxyInit.image.version}} +imagePullPolicy: {{.Values.proxyInit.image.pullPolicy | default .Values.imagePullPolicy}} +name: linkerd-init +{{ include "partials.resources" .Values.proxy.resources }} +securityContext: + {{- if or .Values.proxyInit.closeWaitTimeoutSecs .Values.proxyInit.privileged }} + allowPrivilegeEscalation: true + {{- else }} + allowPrivilegeEscalation: false + {{- end }} + capabilities: + add: + - NET_ADMIN + - NET_RAW + {{- if .Values.proxyInit.capabilities -}} + {{- if .Values.proxyInit.capabilities.add }} + {{- toYaml .Values.proxyInit.capabilities.add | trim | nindent 4 }} + {{- end }} + {{- if .Values.proxyInit.capabilities.drop -}} + {{- include "partials.proxy-init.capabilities.drop" . | nindent 4 -}} + {{- end }} + {{- end }} + {{- if or .Values.proxyInit.closeWaitTimeoutSecs .Values.proxyInit.privileged }} + privileged: true + {{- else }} + privileged: false + {{- end }} + {{- if .Values.proxyInit.runAsRoot }} + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + {{- else }} + runAsNonRoot: true + runAsUser: {{ .Values.proxyInit.runAsUser | int | eq 0 | ternary 65534 .Values.proxyInit.runAsUser }} + runAsGroup: {{ .Values.proxyInit.runAsGroup | int | eq 0 | ternary 65534 .Values.proxyInit.runAsGroup }} + {{- end }} + readOnlyRootFilesystem: true + seccompProfile: + type: RuntimeDefault +terminationMessagePolicy: FallbackToLogsOnError +{{- if or (not .Values.cniEnabled) .Values.proxyInit.saMountPath }} +volumeMounts: +{{- end -}} +{{- if not .Values.cniEnabled }} +- mountPath: {{.Values.proxyInit.xtMountPath.mountPath}} + name: {{.Values.proxyInit.xtMountPath.name}} +{{- end -}} +{{- if .Values.proxyInit.saMountPath }} +- mountPath: {{.Values.proxyInit.saMountPath.mountPath}} + name: {{.Values.proxyInit.saMountPath.name}} + readOnly: {{.Values.proxyInit.saMountPath.readOnly}} +{{- end -}} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy.tpl new file mode 100644 index 0000000000..4dcf12dee2 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_proxy.tpl @@ -0,0 +1,271 @@ +{{ define "partials.proxy" -}} +{{ if and .Values.proxy.nativeSidecar .Values.proxy.waitBeforeExitSeconds }} +{{ fail "proxy.nativeSidecar and waitBeforeExitSeconds cannot be used simultaneously" }} +{{- end }} +{{- if not (has .Values.proxy.logHTTPHeaders (list "insecure" "off" "")) }} +{{- fail "logHTTPHeaders must be one of: insecure | off" }} +{{- end }} +{{- $trustDomain := (.Values.identityTrustDomain | default .Values.clusterDomain) -}} +env: +- name: _pod_name + valueFrom: + fieldRef: + fieldPath: metadata.name +- name: _pod_ns + valueFrom: + fieldRef: + fieldPath: metadata.namespace +- name: _pod_nodeName + valueFrom: + fieldRef: + fieldPath: spec.nodeName +{{- if .Values.proxy.cores }} +- name: LINKERD2_PROXY_CORES + value: {{.Values.proxy.cores | quote}} +{{- end }} +{{ if .Values.proxy.requireIdentityOnInboundPorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_REQUIRE_IDENTITY + value: {{.Values.proxy.requireIdentityOnInboundPorts | quote}} +{{ end -}} +{{ if .Values.proxy.requireTLSOnInboundPorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_REQUIRE_TLS + value: {{.Values.proxy.requireTLSOnInboundPorts | quote}} +{{ end -}} +- name: LINKERD2_PROXY_SHUTDOWN_ENDPOINT_ENABLED + value: {{.Values.proxy.enableShutdownEndpoint | quote}} +- name: LINKERD2_PROXY_LOG + value: "{{.Values.proxy.logLevel}}{{ if not (eq .Values.proxy.logHTTPHeaders "insecure") }},[{headers}]=off,[{request}]=off{{ end }}" +- name: LINKERD2_PROXY_LOG_FORMAT + value: {{.Values.proxy.logFormat | quote}} +- name: LINKERD2_PROXY_DESTINATION_SVC_ADDR + value: {{ternary "localhost.:8086" (printf "linkerd-dst-headless.%s.svc.%s.:8086" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-destination")}} +- name: LINKERD2_PROXY_DESTINATION_PROFILE_NETWORKS + value: {{.Values.clusterNetworks | quote}} +- name: LINKERD2_PROXY_POLICY_SVC_ADDR + value: {{ternary "localhost.:8090" (printf "linkerd-policy.%s.svc.%s.:8090" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-destination")}} +- name: LINKERD2_PROXY_POLICY_WORKLOAD + value: | + {"ns":"$(_pod_ns)", "pod":"$(_pod_name)"} +- name: LINKERD2_PROXY_INBOUND_DEFAULT_POLICY + value: {{.Values.proxy.defaultInboundPolicy}} +- name: LINKERD2_PROXY_POLICY_CLUSTER_NETWORKS + value: {{.Values.clusterNetworks | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_INITIAL_TIMEOUT + value: {{((.Values.proxy.control).streams).initialTimeout | default "" | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_IDLE_TIMEOUT + value: {{((.Values.proxy.control).streams).idleTimeout | default "" | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_LIFETIME + value: {{((.Values.proxy.control).streams).lifetime | default "" | quote}} +{{ if .Values.proxy.inboundConnectTimeout -}} +- name: LINKERD2_PROXY_INBOUND_CONNECT_TIMEOUT + value: {{.Values.proxy.inboundConnectTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.outboundConnectTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_TIMEOUT + value: {{.Values.proxy.outboundConnectTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.outboundDiscoveryCacheUnusedTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_DISCOVERY_IDLE_TIMEOUT + value: {{.Values.proxy.outboundDiscoveryCacheUnusedTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.inboundDiscoveryCacheUnusedTimeout -}} +- name: LINKERD2_PROXY_INBOUND_DISCOVERY_IDLE_TIMEOUT + value: {{.Values.proxy.inboundDiscoveryCacheUnusedTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.disableOutboundProtocolDetectTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_DETECT_TIMEOUT + value: "365d" +{{ end -}} +{{ if .Values.proxy.disableInboundProtocolDetectTimeout -}} +- name: LINKERD2_PROXY_INBOUND_DETECT_TIMEOUT + value: "365d" +{{ end -}} +- name: LINKERD2_PROXY_CONTROL_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.control}}" +- name: LINKERD2_PROXY_ADMIN_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.admin}}" +{{- /* Deprecated, superseded by LINKERD2_PROXY_OUTBOUND_LISTEN_ADDRS since proxy's v2.228.0 (deployed since edge-24.4.5) */}} +- name: LINKERD2_PROXY_OUTBOUND_LISTEN_ADDR + value: "127.0.0.1:{{.Values.proxy.ports.outbound}}" +- name: LINKERD2_PROXY_OUTBOUND_LISTEN_ADDRS + value: "127.0.0.1:{{.Values.proxy.ports.outbound}}{{ if not .Values.disableIPv6}},[::1]:{{.Values.proxy.ports.outbound}}{{ end }}" +- name: LINKERD2_PROXY_INBOUND_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.inbound}}" +- name: LINKERD2_PROXY_INBOUND_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs +- name: LINKERD2_PROXY_INBOUND_PORTS + value: {{ .Values.proxy.podInboundPorts | quote }} +{{ if .Values.proxy.isGateway -}} +- name: LINKERD2_PROXY_INBOUND_GATEWAY_SUFFIXES + value: {{printf "svc.%s." .Values.clusterDomain}} +{{ end -}} +{{ if .Values.proxy.isIngress -}} +- name: LINKERD2_PROXY_INGRESS_MODE + value: "true" +{{ end -}} +- name: LINKERD2_PROXY_DESTINATION_PROFILE_SUFFIXES + {{- $internalDomain := printf "svc.%s." .Values.clusterDomain }} + value: {{ternary "." $internalDomain .Values.proxy.enableExternalProfiles}} +- name: LINKERD2_PROXY_INBOUND_ACCEPT_KEEPALIVE + value: 10000ms +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_KEEPALIVE + value: 10000ms +- name: LINKERD2_PROXY_INBOUND_ACCEPT_USER_TIMEOUT + value: 30s +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_USER_TIMEOUT + value: 30s +{{- /* Configure inbound and outbound parameters, e.g. for HTTP/2 servers. */}} +{{ range $proxyK, $proxyV := (dict "inbound" .Values.proxy.inbound "outbound" .Values.proxy.outbound) -}} +{{ range $scopeK, $scopeV := $proxyV -}} +{{ range $protoK, $protoV := $scopeV -}} +{{ range $paramK, $paramV := $protoV -}} +- name: LINKERD2_PROXY_{{snakecase $proxyK | upper}}_{{snakecase $scopeK | upper}}_{{snakecase $protoK | upper}}_{{snakecase $paramK | upper}} + value: {{ quote $paramV }} +{{ end -}} +{{ end -}} +{{ end -}} +{{ end -}} +{{ if .Values.proxy.opaquePorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_DISABLE_PROTOCOL_DETECTION + value: {{.Values.proxy.opaquePorts | quote}} +{{ end -}} +- name: LINKERD2_PROXY_DESTINATION_CONTEXT + value: | + {"ns":"$(_pod_ns)", "nodeName":"$(_pod_nodeName)", "pod":"$(_pod_name)"} +- name: _pod_sa + valueFrom: + fieldRef: + fieldPath: spec.serviceAccountName +- name: _l5d_ns + value: {{.Release.Namespace}} +- name: _l5d_trustdomain + value: {{$trustDomain}} +- name: LINKERD2_PROXY_IDENTITY_DIR + value: /var/run/linkerd/identity/end-entity +- name: LINKERD2_PROXY_IDENTITY_TRUST_ANCHORS +{{- /* +Pods in the `linkerd` namespace are not injected by the proxy injector and instead obtain +the trust anchor bundle from the `linkerd-identity-trust-roots` configmap. This should not +be used in other contexts. +*/}} +{{- if .Values.proxy.loadTrustBundleFromConfigMap }} + valueFrom: + configMapKeyRef: + name: linkerd-identity-trust-roots + key: ca-bundle.crt +{{ else }} + value: | + {{- required "Please provide the identity trust anchors" .Values.identityTrustAnchorsPEM | trim | nindent 4 }} +{{ end -}} +- name: LINKERD2_PROXY_IDENTITY_TOKEN_FILE +{{- if .Values.identity.serviceAccountTokenProjection }} + value: /var/run/secrets/tokens/linkerd-identity-token +{{ else }} + value: /var/run/secrets/kubernetes.io/serviceaccount/token +{{ end -}} +- name: LINKERD2_PROXY_IDENTITY_SVC_ADDR + value: {{ternary "localhost.:8080" (printf "linkerd-identity-headless.%s.svc.%s.:8080" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-identity")}} +- name: LINKERD2_PROXY_IDENTITY_LOCAL_NAME + value: $(_pod_sa).$(_pod_ns).serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_IDENTITY_SVC_NAME + value: linkerd-identity.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_DESTINATION_SVC_NAME + value: linkerd-destination.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_POLICY_SVC_NAME + value: linkerd-destination.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +{{ if .Values.proxy.accessLog -}} +- name: LINKERD2_PROXY_ACCESS_LOG + value: {{.Values.proxy.accessLog | quote}} +{{ end -}} +{{ if .Values.proxy.shutdownGracePeriod -}} +- name: LINKERD2_PROXY_SHUTDOWN_GRACE_PERIOD + value: {{.Values.proxy.shutdownGracePeriod | quote}} +{{ end -}} +{{ if .Values.proxy.additionalEnv -}} +{{ toYaml .Values.proxy.additionalEnv }} +{{ end -}} +{{ if .Values.proxy.experimentalEnv -}} +{{ toYaml .Values.proxy.experimentalEnv }} +{{ end -}} +image: {{.Values.proxy.image.name}}:{{.Values.proxy.image.version | default .Values.linkerdVersion}} +imagePullPolicy: {{.Values.proxy.image.pullPolicy | default .Values.imagePullPolicy}} +livenessProbe: + httpGet: + path: /live + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.livenessProbe.initialDelaySeconds }} + timeoutSeconds: {{.Values.proxy.livenessProbe.timeoutSeconds }} +name: linkerd-proxy +ports: +- containerPort: {{.Values.proxy.ports.inbound}} + name: linkerd-proxy +- containerPort: {{.Values.proxy.ports.admin}} + name: linkerd-admin +readinessProbe: + httpGet: + path: /ready + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.readinessProbe.initialDelaySeconds }} + timeoutSeconds: {{.Values.proxy.readinessProbe.timeoutSeconds }} +{{- if and .Values.proxy.nativeSidecar .Values.proxy.await }} +startupProbe: + httpGet: + path: /ready + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.startupProbe.initialDelaySeconds}} + periodSeconds: {{.Values.proxy.startupProbe.periodSeconds}} + failureThreshold: {{.Values.proxy.startupProbe.failureThreshold}} +{{- end }} +{{- if .Values.proxy.resources }} +{{ include "partials.resources" .Values.proxy.resources }} +{{- end }} +securityContext: + allowPrivilegeEscalation: false + {{- if .Values.proxy.capabilities -}} + {{- include "partials.proxy.capabilities" . | nindent 2 -}} + {{- end }} + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.proxy.uid}} +{{- if ge (int .Values.proxy.gid) 0 }} + runAsGroup: {{.Values.proxy.gid}} +{{- end }} + seccompProfile: + type: RuntimeDefault +terminationMessagePolicy: FallbackToLogsOnError +{{- if and (not .Values.proxy.nativeSidecar) (or .Values.proxy.await .Values.proxy.waitBeforeExitSeconds) }} +lifecycle: +{{- if .Values.proxy.await }} + postStart: + exec: + command: + - /usr/lib/linkerd/linkerd-await + - --timeout=2m + - --port={{.Values.proxy.ports.admin}} +{{- end }} +{{- if .Values.proxy.waitBeforeExitSeconds }} + preStop: + exec: + command: + - /bin/sleep + - {{.Values.proxy.waitBeforeExitSeconds | quote}} +{{- end }} +{{- end }} +volumeMounts: +- mountPath: /var/run/linkerd/identity/end-entity + name: linkerd-identity-end-entity +{{- if .Values.identity.serviceAccountTokenProjection }} +- mountPath: /var/run/secrets/tokens + name: linkerd-identity-token +{{- end }} +{{- if .Values.proxy.saMountPath }} +- mountPath: {{.Values.proxy.saMountPath.mountPath}} + name: {{.Values.proxy.saMountPath.name}} + readOnly: {{.Values.proxy.saMountPath.readOnly}} +{{- end -}} +{{- if .Values.proxy.nativeSidecar }} +restartPolicy: Always +{{- end -}} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_pull-secrets.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_pull-secrets.tpl new file mode 100644 index 0000000000..0c9aa4f01c --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_pull-secrets.tpl @@ -0,0 +1,6 @@ +{{- define "partials.image-pull-secrets"}} +{{- if . }} +imagePullSecrets: +{{ toYaml . | indent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_resources.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_resources.tpl new file mode 100644 index 0000000000..1fd6789fd7 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_resources.tpl @@ -0,0 +1,28 @@ +{{- define "partials.resources" -}} +{{- $ephemeralStorage := index . "ephemeral-storage" -}} +resources: + {{- if or (.cpu).limit (.memory).limit ($ephemeralStorage).limit }} + limits: + {{- with (.cpu).limit }} + cpu: {{. | quote}} + {{- end }} + {{- with (.memory).limit }} + memory: {{. | quote}} + {{- end }} + {{- with ($ephemeralStorage).limit }} + ephemeral-storage: {{. | quote}} + {{- end }} + {{- end }} + {{- if or (.cpu).request (.memory).request ($ephemeralStorage).request }} + requests: + {{- with (.cpu).request }} + cpu: {{. | quote}} + {{- end }} + {{- with (.memory).request }} + memory: {{. | quote}} + {{- end }} + {{- with ($ephemeralStorage).request }} + ephemeral-storage: {{. | quote}} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_tolerations.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_tolerations.tpl new file mode 100644 index 0000000000..c2292b1464 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_tolerations.tpl @@ -0,0 +1,4 @@ +{{- define "linkerd.tolerations" -}} +tolerations: +{{ toYaml .Values.tolerations | trim | indent 2 }} +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_trace.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_trace.tpl new file mode 100644 index 0000000000..dee059541f --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_trace.tpl @@ -0,0 +1,5 @@ +{{ define "partials.linkerd.trace" -}} +{{ if .Values.controlPlaneTracing -}} +- -trace-collector=collector.{{.Values.controlPlaneTracingNamespace}}.svc.{{.Values.clusterDomain}}:55678 +{{ end -}} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_validate.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_validate.tpl new file mode 100644 index 0000000000..ba772c2fee --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_validate.tpl @@ -0,0 +1,19 @@ +{{- define "linkerd.webhook.validation" -}} + +{{- if and (.injectCaFrom) (.injectCaFromSecret) -}} +{{- fail "injectCaFrom and injectCaFromSecret cannot both be set" -}} +{{- end -}} + +{{- if and (or (.injectCaFrom) (.injectCaFromSecret)) (.caBundle) -}} +{{- fail "injectCaFrom or injectCaFromSecret cannot be set if providing a caBundle" -}} +{{- end -}} + +{{- if and (.externalSecret) (empty .caBundle) (empty .injectCaFrom) (empty .injectCaFromSecret) -}} +{{- fail "if externalSecret is set, then caBundle, injectCaFrom, or injectCaFromSecret must be set" -}} +{{- end }} + +{{- if and (or .injectCaFrom .injectCaFromSecret .caBundle) (not .externalSecret) -}} +{{- fail "if caBundle, injectCaFrom, or injectCaFromSecret is set, then externalSecret must be set" -}} +{{- end -}} + +{{- end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_volumes.tpl b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_volumes.tpl new file mode 100644 index 0000000000..ecb24cfe63 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/templates/_volumes.tpl @@ -0,0 +1,41 @@ +{{ define "partials.proxy.volumes.identity" -}} +emptyDir: + medium: Memory +name: linkerd-identity-end-entity +{{- end -}} + +{{ define "partials.proxyInit.volumes.xtables" -}} +emptyDir: {} +name: {{ .Values.proxyInit.xtMountPath.name }} +{{- end -}} + +{{- define "partials.proxy.volumes.service-account-token" -}} +name: linkerd-identity-token +projected: + sources: + - serviceAccountToken: + path: linkerd-identity-token + expirationSeconds: 86400 {{- /* # 24 hours */}} + audience: identity.l5d.io +{{- end -}} + +{{- define "partials.volumes.manual-mount-service-account-token" -}} +name: kube-api-access +projected: + defaultMode: 420 + sources: + - serviceAccountToken: + expirationSeconds: 3607 + path: token + - configMap: + items: + - key: ca.crt + path: ca.crt + name: kube-root-ca.crt + - downwardAPI: + items: + - fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + path: namespace +{{- end -}} \ No newline at end of file diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/values.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/charts/partials/values.yaml new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/questions.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/questions.yaml new file mode 100644 index 0000000000..4ae27870a3 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/questions.yaml @@ -0,0 +1,19 @@ +questions: +- variable: identityTrustAnchorsPEM + label: "Trust root certificate (ECDSA)" + description: "Root certificate used to support mTLS connections between meshed pods" + required: true + type: multiline + group: Identity +- variable: identity.issuer.tls.crtPEM + label: "Issuer certificate (ECDSA)" + description: "Intermediate certificate, rooted on identityTrustAnchorsPEM, used to sign the Linkerd proxies' CSR" + required: true + type: multiline + group: Identity +- variable: identity.issuer.tls.keyPEM + label: "Key for the issuer certificate (ECDSA)" + description: "Private key for the certificate entered on crtPEM" + required: true + type: multiline + group: Identity diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/NOTES.txt b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/NOTES.txt new file mode 100644 index 0000000000..4bd1be9fc0 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/NOTES.txt @@ -0,0 +1,19 @@ +The Linkerd control plane was successfully installed 🎉 + +To help you manage your Linkerd service mesh you can install the Linkerd CLI by running: + + curl -sL https://run.linkerd.io/install | sh + +Alternatively, you can download the CLI directly via the Linkerd releases page: + + https://github.com/linkerd/linkerd2/releases/ + +To make sure everything works as expected, run the following: + + linkerd check + +The viz extension can be installed by running: + + helm install linkerd-viz linkerd/linkerd-viz + +Looking for more? Visit https://linkerd.io/2/getting-started/ diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config-rbac.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config-rbac.yaml new file mode 100644 index 0000000000..5f5c34203e --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config-rbac.yaml @@ -0,0 +1,16 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} + name: ext-namespace-metadata-linkerd-config + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["get"] + resourceNames: ["linkerd-config"] diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config.yaml new file mode 100644 index 0000000000..a9cea5f421 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/config.yaml @@ -0,0 +1,39 @@ +--- +kind: ConfigMap +apiVersion: v1 +metadata: + name: linkerd-config + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: controller + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +data: + linkerd-crds-chart-version: linkerd-crds-1.0.0-edge + values: | + {{- $values := deepCopy .Values }} + {{- /* + WARNING! All sensitive or private data such as TLS keys must be removed + here to avoid it being publicly readable. + */ -}} + {{- if kindIs "map" $values.identity.issuer.tls -}} + {{- $_ := unset $values.identity.issuer.tls "keyPEM"}} + {{- end -}} + {{- if kindIs "map" $values.profileValidator -}} + {{- $_ := unset $values.profileValidator "keyPEM"}} + {{- end -}} + {{- if kindIs "map" $values.proxyInjector -}} + {{- $_ := unset $values.proxyInjector "keyPEM"}} + {{- end -}} + {{- if kindIs "map" $values.policyValidator -}} + {{- $_ := unset $values.policyValidator "keyPEM"}} + {{- end -}} + {{- if (empty $values.identityTrustDomain) -}} + {{- $_ := set $values "identityTrustDomain" $values.clusterDomain}} + {{- end -}} + {{- $_ := unset $values "partials"}} + {{- $_ := unset $values "configs"}} + {{- $_ := unset $values "stage"}} + {{- toYaml $values | trim | nindent 4 }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination-rbac.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination-rbac.yaml new file mode 100644 index 0000000000..83f96bca02 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination-rbac.yaml @@ -0,0 +1,339 @@ +--- +### +### Destination Controller Service +### +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-destination + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: ["apps"] + resources: ["replicasets"] + verbs: ["list", "get", "watch"] +- apiGroups: ["batch"] + resources: ["jobs"] + verbs: ["list", "get", "watch"] +- apiGroups: [""] + resources: ["pods", "endpoints", "services", "nodes"] + verbs: ["list", "get", "watch"] +- apiGroups: ["linkerd.io"] + resources: ["serviceprofiles"] + verbs: ["list", "get", "watch"] +- apiGroups: ["workload.linkerd.io"] + resources: ["externalworkloads"] + verbs: ["list", "get", "watch"] +- apiGroups: ["coordination.k8s.io"] + resources: ["leases"] + verbs: ["create", "get", "update", "patch"] + {{- if .Values.enableEndpointSlices }} +- apiGroups: ["discovery.k8s.io"] + resources: ["endpointslices"] + verbs: ["list", "get", "watch", "create", "update", "patch", "delete"] + {{- end }} +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-destination + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: linkerd-{{.Release.Namespace}}-destination +subjects: +- kind: ServiceAccount + name: linkerd-destination + namespace: {{.Release.Namespace}} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: linkerd-destination + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +{{- include "partials.image-pull-secrets" .Values.imagePullSecrets }} +--- +{{- $host := printf "linkerd-sp-validator.%s.svc" .Release.Namespace }} +{{- $ca := genSelfSignedCert $host (list) (list $host) 365 }} +{{- if (not .Values.profileValidator.externalSecret) }} +kind: Secret +apiVersion: v1 +metadata: + name: linkerd-sp-validator-k8s-tls + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +type: kubernetes.io/tls +data: + tls.crt: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.profileValidator.crtPEM)) (empty .Values.profileValidator.crtPEM) }} + tls.key: {{ ternary (b64enc (trim $ca.Key)) (b64enc (trim .Values.profileValidator.keyPEM)) (empty .Values.profileValidator.keyPEM) }} +--- +{{- end }} +{{- include "linkerd.webhook.validation" .Values.profileValidator }} +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: linkerd-sp-validator-webhook-config + {{- if or (.Values.profileValidator.injectCaFrom) (.Values.profileValidator.injectCaFromSecret) }} + annotations: + {{- if .Values.profileValidator.injectCaFrom }} + cert-manager.io/inject-ca-from: {{ .Values.profileValidator.injectCaFrom }} + {{- end }} + {{- if .Values.profileValidator.injectCaFromSecret }} + cert-manager.io/inject-ca-from-secret: {{ .Values.profileValidator.injectCaFromSecret }} + {{- end }} + {{- end }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +webhooks: +- name: linkerd-sp-validator.linkerd.io + namespaceSelector: + {{- toYaml .Values.profileValidator.namespaceSelector | trim | nindent 4 }} + clientConfig: + service: + name: linkerd-sp-validator + namespace: {{ .Release.Namespace }} + path: "/" + {{- if and (empty .Values.profileValidator.injectCaFrom) (empty .Values.profileValidator.injectCaFromSecret) }} + caBundle: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.profileValidator.caBundle)) (empty .Values.profileValidator.caBundle) }} + {{- end }} + failurePolicy: {{.Values.webhookFailurePolicy}} + admissionReviewVersions: ["v1", "v1beta1"] + rules: + - operations: ["CREATE", "UPDATE"] + apiGroups: ["linkerd.io"] + apiVersions: ["v1alpha1", "v1alpha2"] + resources: ["serviceprofiles"] + sideEffects: None +--- +{{- $host := printf "linkerd-policy-validator.%s.svc" .Release.Namespace }} +{{- $ca := genSelfSignedCert $host (list) (list $host) 365 }} +{{- if (not .Values.policyValidator.externalSecret) }} +kind: Secret +apiVersion: v1 +metadata: + name: linkerd-policy-validator-k8s-tls + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +type: kubernetes.io/tls +data: + tls.crt: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.policyValidator.crtPEM)) (empty .Values.policyValidator.crtPEM) }} + tls.key: {{ ternary (b64enc (trim $ca.Key)) (b64enc (trim .Values.policyValidator.keyPEM)) (empty .Values.policyValidator.keyPEM) }} +--- +{{- end }} +{{- include "linkerd.webhook.validation" .Values.policyValidator }} +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: linkerd-policy-validator-webhook-config + {{- if or (.Values.policyValidator.injectCaFrom) (.Values.policyValidator.injectCaFromSecret) }} + annotations: + {{- if .Values.policyValidator.injectCaFrom }} + cert-manager.io/inject-ca-from: {{ .Values.policyValidator.injectCaFrom }} + {{- end }} + {{- if .Values.policyValidator.injectCaFromSecret }} + cert-manager.io/inject-ca-from-secret: {{ .Values.policyValidator.injectCaFromSecret }} + {{- end }} + {{- end }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +webhooks: +- name: linkerd-policy-validator.linkerd.io + namespaceSelector: + {{- toYaml .Values.policyValidator.namespaceSelector | trim | nindent 4 }} + clientConfig: + service: + name: linkerd-policy-validator + namespace: {{ .Release.Namespace }} + path: "/" + {{- if and (empty .Values.policyValidator.injectCaFrom) (empty .Values.policyValidator.injectCaFromSecret) }} + caBundle: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.policyValidator.caBundle)) (empty .Values.policyValidator.caBundle) }} + {{- end }} + failurePolicy: {{.Values.webhookFailurePolicy}} + admissionReviewVersions: ["v1", "v1beta1"] + rules: + - operations: ["CREATE", "UPDATE"] + apiGroups: ["policy.linkerd.io"] + apiVersions: ["*"] + resources: + - authorizationpolicies + - httplocalratelimitpolicies + - httproutes + - networkauthentications + - meshtlsauthentications + - serverauthorizations + - servers + - egressnetworks + - operations: ["CREATE", "UPDATE"] + apiGroups: ["gateway.networking.k8s.io"] + apiVersions: ["*"] + resources: + - httproutes + - grpcroutes + - tlsroutes + - tcproutes + sideEffects: None +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: linkerd-policy + labels: + app.kubernetes.io/part-of: Linkerd + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - apiGroups: + - apps + resources: + - deployments + verbs: + - get + - apiGroups: + - policy.linkerd.io + resources: + - authorizationpolicies + - httplocalratelimitpolicies + - httproutes + - meshtlsauthentications + - networkauthentications + - servers + - serverauthorizations + - egressnetworks + verbs: + - get + - list + - watch + - apiGroups: + - gateway.networking.k8s.io + resources: + - httproutes + - grpcroutes + - tlsroutes + - tcproutes + verbs: + - get + - list + - watch + - apiGroups: + - policy.linkerd.io + resources: + - httproutes/status + - httplocalratelimitpolicies/status + - egressnetworks/status + verbs: + - patch + - apiGroups: + - gateway.networking.k8s.io + resources: + - httproutes/status + - grpcroutes/status + - tlsroutes/status + - tcproutes/status + verbs: + - patch + - apiGroups: + - workload.linkerd.io + resources: + - externalworkloads + verbs: + - get + - list + - watch + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - create + - get + - patch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: linkerd-destination-policy + labels: + app.kubernetes.io/part-of: Linkerd + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: linkerd-policy +subjects: + - kind: ServiceAccount + name: linkerd-destination + namespace: {{.Release.Namespace}} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: remote-discovery + namespace: {{.Release.Namespace}} + labels: + app.kubernetes.io/part-of: Linkerd + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - list + - watch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: linkerd-destination-remote-discovery + namespace: {{.Release.Namespace}} + labels: + app.kubernetes.io/part-of: Linkerd + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: remote-discovery +subjects: + - kind: ServiceAccount + name: linkerd-destination + namespace: {{.Release.Namespace}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination.yaml new file mode 100644 index 0000000000..41395c21a8 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/destination.yaml @@ -0,0 +1,448 @@ +--- +### +### Destination Controller Service +### +kind: Service +apiVersion: v1 +metadata: + name: linkerd-dst + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + type: ClusterIP + selector: + linkerd.io/control-plane-component: destination + ports: + - name: grpc + port: 8086 + targetPort: 8086 +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-dst-headless + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + clusterIP: None + selector: + linkerd.io/control-plane-component: destination + ports: + - name: grpc + port: 8086 + targetPort: 8086 +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-sp-validator + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + type: ClusterIP + selector: + linkerd.io/control-plane-component: destination + ports: + - name: sp-validator + port: 443 + targetPort: sp-validator +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-policy + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + clusterIP: None + selector: + linkerd.io/control-plane-component: destination + ports: + - name: grpc + port: 8090 + targetPort: 8090 +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-policy-validator + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + type: ClusterIP + selector: + linkerd.io/control-plane-component: destination + ports: + - name: policy-https + port: 443 + targetPort: policy-https +{{- if .Values.enablePodDisruptionBudget }} +--- +kind: PodDisruptionBudget +apiVersion: policy/v1 +metadata: + name: linkerd-dst + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + maxUnavailable: {{ .Values.controller.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + linkerd.io/control-plane-component: destination +{{- end }} +--- +{{- $tree := deepCopy . }} +{{ $_ := set $tree.Values.proxy "workloadKind" "deployment" -}} +{{ $_ := set $tree.Values.proxy "component" "linkerd-destination" -}} +{{ $_ := set $tree.Values.proxy "waitBeforeExitSeconds" 0 -}} +{{- if not (empty .Values.destinationProxyResources) }} +{{- $c := dig "cores" .Values.proxy.cores .Values.destinationProxyResources }} +{{- $_ := set $tree.Values.proxy "cores" $c }} +{{- $r := merge .Values.destinationProxyResources .Values.proxy.resources }} +{{- $_ := set $tree.Values.proxy "resources" $r }} +{{- end }} +apiVersion: apps/v1 +kind: Deployment +metadata: + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + app.kubernetes.io/name: destination + app.kubernetes.io/part-of: Linkerd + app.kubernetes.io/version: {{.Values.linkerdVersion}} + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + name: linkerd-destination + namespace: {{ .Release.Namespace }} +spec: + replicas: {{.Values.controllerReplicas}} + revisionHistoryLimit: {{.Values.revisionHistoryLimit}} + selector: + matchLabels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- include "partials.proxy.labels" $tree.Values.proxy | nindent 6}} + {{- if .Values.deploymentStrategy }} + strategy: + {{- with .Values.deploymentStrategy }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- end }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/destination-rbac.yaml") . | sha256sum }} + {{ include "partials.annotations.created-by" . }} + {{- include "partials.proxy.annotations" . | nindent 8}} + {{- with .Values.podAnnotations }}{{ toYaml . | trim | nindent 8 }}{{- end }} + config.linkerd.io/default-inbound-policy: "all-unauthenticated" + labels: + linkerd.io/control-plane-component: destination + linkerd.io/control-plane-ns: {{.Release.Namespace}} + linkerd.io/workload-ns: {{.Release.Namespace}} + {{- include "partials.proxy.labels" $tree.Values.proxy | nindent 8}} + {{- with .Values.podLabels }}{{ toYaml . | trim | nindent 8 }}{{- end }} + spec: + {{- with .Values.runtimeClassName }} + runtimeClassName: {{ . | quote }} + {{- end }} + {{- if .Values.tolerations -}} + {{- include "linkerd.tolerations" . | nindent 6 }} + {{- end -}} + {{- include "linkerd.node-selector" . | nindent 6 }} + {{- $_ := set $tree "component" "destination" -}} + {{- include "linkerd.affinity" $tree | nindent 6 }} + automountServiceAccountToken: false + containers: + {{- $_ := set $tree.Values.proxy "await" $tree.Values.proxy.await }} + {{- $_ := set $tree.Values.proxy "loadTrustBundleFromConfigMap" true }} + {{- $_ := set $tree.Values.proxy "podInboundPorts" "8086,8090,8443,9443,9990,9996,9997" }} + {{- $_ := set $tree.Values.proxy "outboundDiscoveryCacheUnusedTimeout" "5s" }} + {{- $_ := set $tree.Values.proxy "inboundDiscoveryCacheUnusedTimeout" "90s" }} + {{- /* + The pod needs to accept webhook traffic, and we can't rely on that originating in the + cluster network. + */}} + {{- $_ := set $tree.Values.proxy "defaultInboundPolicy" "all-unauthenticated" }} + {{- $_ := set $tree.Values.proxy "capabilities" (dict "drop" (list "ALL")) }} + {{- if not $tree.Values.proxy.nativeSidecar }} + - {{- include "partials.proxy" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{- end }} + - args: + - destination + - -addr=:8086 + - -controller-namespace={{.Release.Namespace}} + - -enable-h2-upgrade={{.Values.enableH2Upgrade}} + - -log-level={{.Values.controllerLogLevel}} + - -log-format={{.Values.controllerLogFormat}} + - -enable-endpoint-slices={{.Values.enableEndpointSlices}} + - -cluster-domain={{.Values.clusterDomain}} + - -identity-trust-domain={{.Values.identityTrustDomain | default .Values.clusterDomain}} + - -default-opaque-ports={{.Values.proxy.opaquePorts}} + - -enable-ipv6={{not .Values.disableIPv6}} + - -enable-pprof={{.Values.enablePprof | default false}} + {{- if (.Values.destinationController).meshedHttp2ClientProtobuf }} + - --meshed-http2-client-params={{ toJson .Values.destinationController.meshedHttp2ClientProtobuf }} + {{- end }} + {{- range (.Values.destinationController).additionalArgs }} + - {{ . }} + {{- end }} + {{- range (.Values.destinationController).experimentalArgs }} + - {{ . }} + {{- end }} + {{- if or (.Values.destinationController).additionalEnv (.Values.destinationController).experimentalEnv }} + env: + {{- with (.Values.destinationController).additionalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- with (.Values.destinationController).experimentalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- end }} + {{- include "partials.linkerd.trace" . | nindent 8 -}} + image: {{.Values.controllerImage}}:{{.Values.controllerImageVersion | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.imagePullPolicy}} + livenessProbe: + httpGet: + path: /ping + port: 9996 + initialDelaySeconds: 10 + {{- with (.Values.destinationController.livenessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + name: destination + ports: + - containerPort: 8086 + name: grpc + - containerPort: 9996 + name: admin-http + readinessProbe: + failureThreshold: 7 + httpGet: + path: /ready + port: 9996 + {{- with (.Values.destinationController.readinessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + {{- if .Values.destinationResources -}} + {{- include "partials.resources" .Values.destinationResources | nindent 8 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + - args: + - sp-validator + - -log-level={{.Values.controllerLogLevel}} + - -log-format={{.Values.controllerLogFormat}} + - -enable-pprof={{.Values.enablePprof | default false}} + {{- if or (.Values.spValidator).additionalEnv (.Values.spValidator).experimentalEnv }} + env: + {{- with (.Values.spValidator).additionalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- with (.Values.spValidator).experimentalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- end }} + image: {{.Values.controllerImage}}:{{.Values.controllerImageVersion | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.imagePullPolicy}} + livenessProbe: + httpGet: + path: /ping + port: 9997 + initialDelaySeconds: 10 + {{- with ((.Values.spValidator).livenessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + name: sp-validator + ports: + - containerPort: 8443 + name: sp-validator + - containerPort: 9997 + name: admin-http + readinessProbe: + failureThreshold: 7 + httpGet: + path: /ready + port: 9997 + {{- with ((.Values.spValidator).readinessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + {{- if .Values.spValidatorResources -}} + {{- include "partials.resources" .Values.spValidatorResources | nindent 8 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/linkerd/tls + name: sp-tls + readOnly: true + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + - args: + - --admin-addr={{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:9990 + - --control-plane-namespace={{.Release.Namespace}} + - --grpc-addr={{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:8090 + - --server-addr={{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:9443 + - --server-tls-key=/var/run/linkerd/tls/tls.key + - --server-tls-certs=/var/run/linkerd/tls/tls.crt + - --cluster-networks={{.Values.clusterNetworks}} + - --identity-domain={{.Values.identityTrustDomain | default .Values.clusterDomain}} + - --cluster-domain={{.Values.clusterDomain}} + - --default-policy={{.Values.proxy.defaultInboundPolicy}} + - --log-level={{.Values.policyController.logLevel | default "linkerd=info,warn"}} + - --log-format={{.Values.controllerLogFormat}} + - --default-opaque-ports={{.Values.proxy.opaquePorts}} + - --global-egress-network-namespace={{.Values.egress.globalEgressNetworkNamespace}} + {{- if .Values.policyController.probeNetworks }} + - --probe-networks={{.Values.policyController.probeNetworks | join ","}} + {{- end}} + {{- range .Values.policyController.additionalArgs }} + - {{ . }} + {{- end }} + {{- range .Values.policyController.experimentalArgs }} + - {{ . }} + {{- end }} + image: {{.Values.policyController.image.name}}:{{.Values.policyController.image.version | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.policyController.image.pullPolicy | default .Values.imagePullPolicy}} + livenessProbe: + httpGet: + path: /live + port: admin-http + {{- with (.Values.policyController.livenessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + name: policy + ports: + - containerPort: 8090 + name: grpc + - containerPort: 9990 + name: admin-http + - containerPort: 9443 + name: policy-https + readinessProbe: + failureThreshold: 7 + httpGet: + path: /ready + port: admin-http + initialDelaySeconds: 10 + {{- with (.Values.policyController.readinessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + {{- if .Values.policyController.resources }} + {{- include "partials.resources" .Values.policyController.resources | nindent 8 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/linkerd/tls + name: policy-tls + readOnly: true + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + initContainers: + {{ if .Values.cniEnabled -}} + - {{- include "partials.network-validator" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ else -}} + {{- /* + The destination controller needs to connect to the Kubernetes API before the proxy is able + to proxy requests, so we always skip these connections. + */}} + {{- $_ := set $tree.Values.proxyInit "ignoreOutboundPorts" .Values.proxyInit.kubeAPIServerPorts -}} + - {{- include "partials.proxy-init" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{- if $tree.Values.proxy.nativeSidecar }} + {{- $_ := set $tree.Values.proxy "startupProbeInitialDelaySeconds" 35 }} + {{- $_ := set $tree.Values.proxy "startupProbePeriodSeconds" 5 }} + {{- $_ := set $tree.Values.proxy "startupProbeFailureThreshold" 20 }} + - {{- include "partials.proxy" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{- if .Values.priorityClassName -}} + priorityClassName: {{ .Values.priorityClassName }} + {{ end -}} + securityContext: + seccompProfile: + type: RuntimeDefault + serviceAccountName: linkerd-destination + volumes: + - name: sp-tls + secret: + secretName: linkerd-sp-validator-k8s-tls + - name: policy-tls + secret: + secretName: linkerd-policy-validator-k8s-tls + - {{- include "partials.volumes.manual-mount-service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ if not .Values.cniEnabled -}} + - {{- include "partials.proxyInit.volumes.xtables" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{if .Values.identity.serviceAccountTokenProjection -}} + - {{- include "partials.proxy.volumes.service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + - {{- include "partials.proxy.volumes.identity" . | indent 8 | trimPrefix (repeat 7 " ") }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat-rbac.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat-rbac.yaml new file mode 100644 index 0000000000..7b127543f4 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat-rbac.yaml @@ -0,0 +1,78 @@ +{{ if not .Values.disableHeartBeat -}} +--- +### +### Heartbeat RBAC +### +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: linkerd-heartbeat + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["get"] + resourceNames: ["linkerd-config"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: linkerd-heartbeat + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + kind: Role + name: linkerd-heartbeat + apiGroup: rbac.authorization.k8s.io +subjects: +- kind: ServiceAccount + name: linkerd-heartbeat + namespace: {{.Release.Namespace}} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: linkerd-heartbeat + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: [""] + resources: ["namespaces"] + verbs: ["list"] +- apiGroups: ["linkerd.io"] + resources: ["serviceprofiles"] + verbs: ["list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: linkerd-heartbeat + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + kind: ClusterRole + name: linkerd-heartbeat + apiGroup: rbac.authorization.k8s.io +subjects: +- kind: ServiceAccount + name: linkerd-heartbeat + namespace: {{.Release.Namespace}} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: linkerd-heartbeat + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: heartbeat + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +{{- include "partials.image-pull-secrets" .Values.imagePullSecrets }} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat.yaml new file mode 100644 index 0000000000..1f42493849 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/heartbeat.yaml @@ -0,0 +1,101 @@ +{{ if not .Values.disableHeartBeat -}} +--- +### +### Heartbeat +### +apiVersion: batch/v1 +kind: CronJob +metadata: + name: linkerd-heartbeat + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/name: heartbeat + app.kubernetes.io/part-of: Linkerd + app.kubernetes.io/version: {{.Values.linkerdVersion}} + linkerd.io/control-plane-component: heartbeat + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + concurrencyPolicy: Replace + {{ if .Values.heartbeatSchedule -}} + schedule: "{{.Values.heartbeatSchedule}}" + {{ else -}} + schedule: "{{ dateInZone "04 15 * * *" (now | mustDateModify "+10m") "UTC"}}" + {{ end -}} + successfulJobsHistoryLimit: 0 + jobTemplate: + spec: + template: + metadata: + labels: + linkerd.io/control-plane-component: heartbeat + linkerd.io/workload-ns: {{.Release.Namespace}} + {{- with .Values.podLabels }}{{ toYaml . | trim | nindent 12 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} + {{- with .Values.podAnnotations }}{{ toYaml . | trim | nindent 12 }}{{- end }} + spec: + {{- if .Values.priorityClassName }} + priorityClassName: {{ .Values.priorityClassName }} + {{- end -}} + {{- with .Values.runtimeClassName }} + runtimeClassName: {{ . | quote }} + {{- end }} + {{- if .Values.tolerations -}} + {{- include "linkerd.tolerations" . | nindent 10 }} + {{- end -}} + {{- include "linkerd.node-selector" . | nindent 10 }} + securityContext: + seccompProfile: + type: RuntimeDefault + serviceAccountName: linkerd-heartbeat + restartPolicy: Never + automountServiceAccountToken: false + containers: + - name: heartbeat + image: {{.Values.controllerImage}}:{{.Values.controllerImageVersion | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.imagePullPolicy}} + env: + - name: LINKERD_DISABLED + value: "the heartbeat controller does not use the proxy" + {{- with (.Values.heartbeat).additionalEnv }} + {{- toYaml . | nindent 12 -}} + {{- end }} + {{- with (.Values.heartbeat).experimentalEnv }} + {{- toYaml . | nindent 12 -}} + {{- end }} + args: + - "heartbeat" + - "-controller-namespace={{.Release.Namespace}}" + - "-log-level={{.Values.controllerLogLevel}}" + - "-log-format={{.Values.controllerLogFormat}}" + {{- if .Values.prometheusUrl }} + - "-prometheus-url={{.Values.prometheusUrl}}" + {{- else }} + - "-prometheus-url=http://prometheus.linkerd-viz.svc.{{.Values.clusterDomain}}:9090" + {{- end }} + {{- if .Values.heartbeatResources -}} + {{- include "partials.resources" .Values.heartbeatResources | nindent 12 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + volumes: + - {{- include "partials.volumes.manual-mount-service-account-token" . | indent 12 | trimPrefix (repeat 11 " ") }} +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity-rbac.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity-rbac.yaml new file mode 100644 index 0000000000..6efdb4e104 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity-rbac.yaml @@ -0,0 +1,49 @@ +--- +### +### Identity Controller Service RBAC +### +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-identity + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: ["authentication.k8s.io"] + resources: ["tokenreviews"] + verbs: ["create"] +# TODO(ver) Restrict this to the Linkerd namespace. See +# https://github.com/linkerd/linkerd2/issues/9367 +- apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-identity + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: linkerd-{{.Release.Namespace}}-identity +subjects: +- kind: ServiceAccount + name: linkerd-identity + namespace: {{.Release.Namespace}} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: linkerd-identity + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +{{- include "partials.image-pull-secrets" .Values.imagePullSecrets }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity.yaml new file mode 100644 index 0000000000..960cd93399 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/identity.yaml @@ -0,0 +1,277 @@ +{{if .Values.identity -}} +--- +### +### Identity Controller Service +### +{{ if and (.Values.identity.issuer) (eq .Values.identity.issuer.scheme "linkerd.io/tls") -}} +kind: Secret +apiVersion: v1 +metadata: + name: linkerd-identity-issuer + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +data: + crt.pem: {{b64enc (required "Please provide the identity issuer certificate" .Values.identity.issuer.tls.crtPEM | trim)}} + key.pem: {{b64enc (required "Please provide the identity issue private key" .Values.identity.issuer.tls.keyPEM | trim)}} +--- +{{- end}} +{{ if not (.Values.identity.externalCA) -}} +kind: ConfigMap +apiVersion: v1 +metadata: + name: linkerd-identity-trust-roots + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +data: + ca-bundle.crt: |-{{.Values.identityTrustAnchorsPEM | trim | nindent 4}} +--- +{{- end}} +kind: Service +apiVersion: v1 +metadata: + name: linkerd-identity + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + type: ClusterIP + selector: + linkerd.io/control-plane-component: identity + ports: + - name: grpc + port: 8080 + targetPort: 8080 +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-identity-headless + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + clusterIP: None + selector: + linkerd.io/control-plane-component: identity + ports: + - name: grpc + port: 8080 + targetPort: 8080 +--- +{{- if .Values.enablePodDisruptionBudget }} +kind: PodDisruptionBudget +apiVersion: policy/v1 +metadata: + name: linkerd-identity + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + maxUnavailable: {{ .Values.controller.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + linkerd.io/control-plane-component: identity +--- +{{- end }} +{{- $tree := deepCopy . }} +{{ $_ := set $tree.Values.proxy "workloadKind" "deployment" -}} +{{ $_ := set $tree.Values.proxy "component" "linkerd-identity" -}} +{{ $_ := set $tree.Values.proxy "waitBeforeExitSeconds" 0 -}} +{{- if not (empty .Values.identityProxyResources) }} +{{- $c := dig "cores" .Values.proxy.cores .Values.identityProxyResources }} +{{- $_ := set $tree.Values.proxy "cores" $c }} +{{- $r := merge .Values.identityProxyResources .Values.proxy.resources }} +{{- $_ := set $tree.Values.proxy "resources" $r }} +{{- end }} +apiVersion: apps/v1 +kind: Deployment +metadata: + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + app.kubernetes.io/name: identity + app.kubernetes.io/part-of: Linkerd + app.kubernetes.io/version: {{.Values.linkerdVersion}} + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + name: linkerd-identity + namespace: {{ .Release.Namespace }} +spec: + replicas: {{.Values.controllerReplicas}} + revisionHistoryLimit: {{.Values.revisionHistoryLimit}} + selector: + matchLabels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- include "partials.proxy.labels" $tree.Values.proxy | nindent 6}} + {{- if .Values.deploymentStrategy }} + strategy: + {{- with .Values.deploymentStrategy }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- end }} + template: + metadata: + annotations: + {{ include "partials.annotations.created-by" . }} + {{- include "partials.proxy.annotations" . | nindent 8}} + {{- with .Values.podAnnotations }}{{ toYaml . | trim | nindent 8 }}{{- end }} + config.linkerd.io/default-inbound-policy: "all-unauthenticated" + labels: + linkerd.io/control-plane-component: identity + linkerd.io/control-plane-ns: {{.Release.Namespace}} + linkerd.io/workload-ns: {{.Release.Namespace}} + {{- include "partials.proxy.labels" $tree.Values.proxy | nindent 8}} + {{- with .Values.podLabels }}{{ toYaml . | trim | nindent 8 }}{{- end }} + spec: + {{- with .Values.runtimeClassName }} + runtimeClassName: {{ . | quote }} + {{- end }} + {{- if .Values.tolerations -}} + {{- include "linkerd.tolerations" . | nindent 6 }} + {{- end -}} + {{- include "linkerd.node-selector" . | nindent 6 }} + {{- $_ := set $tree "component" "identity" -}} + {{- include "linkerd.affinity" $tree | nindent 6 }} + automountServiceAccountToken: false + containers: + - args: + - identity + - -log-level={{.Values.controllerLogLevel}} + - -log-format={{.Values.controllerLogFormat}} + - -controller-namespace={{.Release.Namespace}} + - -identity-trust-domain={{.Values.identityTrustDomain | default .Values.clusterDomain}} + - -identity-issuance-lifetime={{.Values.identity.issuer.issuanceLifetime}} + - -identity-clock-skew-allowance={{.Values.identity.issuer.clockSkewAllowance}} + - -identity-scheme={{.Values.identity.issuer.scheme}} + - -enable-pprof={{.Values.enablePprof | default false}} + - -kube-apiclient-qps={{.Values.identity.kubeAPI.clientQPS}} + - -kube-apiclient-burst={{.Values.identity.kubeAPI.clientBurst}} + {{- include "partials.linkerd.trace" . | nindent 8 -}} + env: + - name: LINKERD_DISABLED + value: "linkerd-await cannot block the identity controller" + {{- with (.Values.identity).additionalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- with (.Values.identity).experimentalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + image: {{.Values.controllerImage}}:{{.Values.controllerImageVersion | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.imagePullPolicy}} + livenessProbe: + httpGet: + path: /ping + port: 9990 + initialDelaySeconds: 10 + {{- with (.Values.identity.livenessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + name: identity + ports: + - containerPort: 8080 + name: grpc + - containerPort: 9990 + name: admin-http + readinessProbe: + failureThreshold: 7 + httpGet: + path: /ready + port: 9990 + {{- with (.Values.identity.readinessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + {{- if .Values.identityResources -}} + {{- include "partials.resources" .Values.identityResources | nindent 8 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/linkerd/identity/issuer + name: identity-issuer + - mountPath: /var/run/linkerd/identity/trust-roots/ + name: trust-roots + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + {{- $_ := set $tree.Values.proxy "await" false }} + {{- $_ := set $tree.Values.proxy "loadTrustBundleFromConfigMap" true }} + {{- $_ := set $tree.Values.proxy "podInboundPorts" "8080,9990" }} + {{- $_ := set $tree.Values.proxy "nativeSidecar" false }} + {{- /* + The identity controller cannot discover policies, so we configure it with defaults that + enforce TLS on the identity service. + */}} + {{- $_ := set $tree.Values.proxy "defaultInboundPolicy" "all-unauthenticated" }} + {{- $_ := set $tree.Values.proxy "requireTLSOnInboundPorts" "8080" }} + {{- $_ := set $tree.Values.proxy "capabilities" (dict "drop" (list "ALL")) }} + {{- $_ := set $tree.Values.proxy "outboundDiscoveryCacheUnusedTimeout" "5s" }} + {{- $_ := set $tree.Values.proxy "inboundDiscoveryCacheUnusedTimeout" "90s" }} + - {{- include "partials.proxy" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + initContainers: + {{ if .Values.cniEnabled -}} + - {{- include "partials.network-validator" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ else -}} + {{- /* + The identity controller needs to connect to the Kubernetes API before the proxy is able to + proxy requests, so we always skip these connections. The identity controller makes no other + outbound connections (so it's not important to persist any other skip ports here) + */}} + {{- $_ := set $tree.Values.proxyInit "ignoreOutboundPorts" .Values.proxyInit.kubeAPIServerPorts -}} + - {{- include "partials.proxy-init" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{- if .Values.priorityClassName -}} + priorityClassName: {{ .Values.priorityClassName }} + {{ end -}} + securityContext: + seccompProfile: + type: RuntimeDefault + serviceAccountName: linkerd-identity + volumes: + - name: identity-issuer + secret: + secretName: linkerd-identity-issuer + - configMap: + name: linkerd-identity-trust-roots + name: trust-roots + - {{- include "partials.volumes.manual-mount-service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ if not .Values.cniEnabled -}} + - {{- include "partials.proxyInit.volumes.xtables" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{if .Values.identity.serviceAccountTokenProjection -}} + - {{- include "partials.proxy.volumes.service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + - {{- include "partials.proxy.volumes.identity" . | indent 8 | trimPrefix (repeat 7 " ") }} +{{end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/namespace.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/namespace.yaml new file mode 100644 index 0000000000..61461c1327 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/namespace.yaml @@ -0,0 +1,18 @@ +{{- if eq .Release.Service "CLI" -}} +--- +### +### Linkerd Namespace +### +kind: Namespace +apiVersion: v1 +metadata: + name: {{ .Release.Namespace }} + annotations: + linkerd.io/inject: disabled + labels: + linkerd.io/is-control-plane: "true" + config.linkerd.io/admission-webhooks: disabled + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- /* linkerd-init requires extended capabilities and so requires priviledged mode */}} + pod-security.kubernetes.io/enforce: {{ ternary "restricted" "privileged" .Values.cniEnabled }} +{{ end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/podmonitor.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/podmonitor.yaml new file mode 100644 index 0000000000..fd2b5d6ceb --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/podmonitor.yaml @@ -0,0 +1,128 @@ +{{- $podMonitor := .Values.podMonitor -}} +{{- if and $podMonitor.enabled $podMonitor.controller.enabled }} +--- +### +### Prometheus Operator PodMonitor for Linkerd control-plane +### +apiVersion: monitoring.coreos.com/v1 +kind: PodMonitor +metadata: + name: "linkerd-controller" + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{ .Release.Namespace }} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- with .Values.podMonitor.labels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + namespaceSelector: {{ tpl .Values.podMonitor.controller.namespaceSelector . | nindent 4 }} + selector: + matchLabels: {} + podMetricsEndpoints: + - interval: {{ $podMonitor.scrapeInterval }} + scrapeTimeout: {{ $podMonitor.scrapeTimeout }} + relabelings: + - sourceLabels: + - __meta_kubernetes_pod_container_port_name + action: keep + regex: admin-http + - sourceLabels: + - __meta_kubernetes_pod_container_name + action: replace + targetLabel: component +{{- end }} +{{- if and $podMonitor.enabled $podMonitor.serviceMirror.enabled }} +--- +### +### Prometheus Operator PodMonitor for Linkerd Service Mirror (multi-cluster) +### +apiVersion: monitoring.coreos.com/v1 +kind: PodMonitor +metadata: + name: "linkerd-service-mirror" + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{ .Release.Namespace }} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- with .Values.podMonitor.labels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + namespaceSelector: + any: true + selector: + matchLabels: {} + podMetricsEndpoints: + - interval: {{ $podMonitor.scrapeInterval }} + scrapeTimeout: {{ $podMonitor.scrapeTimeout }} + relabelings: + - sourceLabels: + - __meta_kubernetes_pod_label_linkerd_io_control_plane_component + - __meta_kubernetes_pod_container_port_name + action: keep + regex: linkerd-service-mirror;admin-http$ + - sourceLabels: + - __meta_kubernetes_pod_container_name + action: replace + targetLabel: component +{{- end }} +{{- if and $podMonitor.enabled $podMonitor.proxy.enabled }} +--- +### +### Prometheus Operator PodMonitor Linkerd data-plane +### +apiVersion: monitoring.coreos.com/v1 +kind: PodMonitor +metadata: + name: "linkerd-proxy" + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{ .Release.Namespace }} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- with .Values.podMonitor.labels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + namespaceSelector: + any: true + selector: + matchLabels: {} + podMetricsEndpoints: + - interval: {{ $podMonitor.scrapeInterval }} + scrapeTimeout: {{ $podMonitor.scrapeTimeout }} + relabelings: + - sourceLabels: + - __meta_kubernetes_pod_container_name + - __meta_kubernetes_pod_container_port_name + - __meta_kubernetes_pod_label_linkerd_io_control_plane_ns + action: keep + regex: ^linkerd-proxy;linkerd-admin;{{ .Release.Namespace }}$ + - sourceLabels: [ __meta_kubernetes_namespace ] + action: replace + targetLabel: namespace + - sourceLabels: [ __meta_kubernetes_pod_name ] + action: replace + targetLabel: pod + - sourceLabels: [ __meta_kubernetes_pod_label_linkerd_io_proxy_job ] + action: replace + targetLabel: k8s_job + - action: labeldrop + regex: __meta_kubernetes_pod_label_linkerd_io_proxy_job + - action: labelmap + regex: __meta_kubernetes_pod_label_linkerd_io_proxy_(.+) + - action: labeldrop + regex: __meta_kubernetes_pod_label_linkerd_io_proxy_(.+) + - action: labelmap + regex: __meta_kubernetes_pod_label_linkerd_io_(.+) + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + replacement: __tmp_pod_label_$1 + - action: labelmap + regex: __tmp_pod_label_linkerd_io_(.+) + replacement: __tmp_pod_label_$1 + - action: labeldrop + regex: __tmp_pod_label_linkerd_io_(.+) + - action: labelmap + regex: __tmp_pod_label_(.+) +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector-rbac.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector-rbac.yaml new file mode 100644 index 0000000000..c2c84c5c17 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector-rbac.yaml @@ -0,0 +1,120 @@ +--- +### +### Proxy Injector RBAC +### +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-proxy-injector + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: [""] + resources: ["events"] + verbs: ["create", "patch"] +- apiGroups: [""] + resources: ["namespaces", "replicationcontrollers"] + verbs: ["list", "get", "watch"] +- apiGroups: [""] + resources: ["pods"] + verbs: ["list", "watch"] +- apiGroups: ["extensions", "apps"] + resources: ["deployments", "replicasets", "daemonsets", "statefulsets"] + verbs: ["list", "get", "watch"] +- apiGroups: ["extensions", "batch"] + resources: ["cronjobs", "jobs"] + verbs: ["list", "get", "watch"] +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: linkerd-{{.Release.Namespace}}-proxy-injector + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +subjects: +- kind: ServiceAccount + name: linkerd-proxy-injector + namespace: {{.Release.Namespace}} + apiGroup: "" +roleRef: + kind: ClusterRole + name: linkerd-{{.Release.Namespace}}-proxy-injector + apiGroup: rbac.authorization.k8s.io +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: linkerd-proxy-injector + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +{{- include "partials.image-pull-secrets" .Values.imagePullSecrets }} +--- +{{- $host := printf "linkerd-proxy-injector.%s.svc" .Release.Namespace }} +{{- $ca := genSelfSignedCert $host (list) (list $host) 365 }} +{{- if (not .Values.proxyInjector.externalSecret) }} +kind: Secret +apiVersion: v1 +metadata: + name: linkerd-proxy-injector-k8s-tls + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +type: kubernetes.io/tls +data: + tls.crt: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.proxyInjector.crtPEM)) (empty .Values.proxyInjector.crtPEM) }} + tls.key: {{ ternary (b64enc (trim $ca.Key)) (b64enc (trim .Values.proxyInjector.keyPEM)) (empty .Values.proxyInjector.keyPEM) }} +--- +{{- end }} +{{- include "linkerd.webhook.validation" .Values.proxyInjector }} +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: linkerd-proxy-injector-webhook-config + {{- if or (.Values.proxyInjector.injectCaFrom) (.Values.proxyInjector.injectCaFromSecret) }} + annotations: + {{- if .Values.proxyInjector.injectCaFrom }} + cert-manager.io/inject-ca-from: {{ .Values.proxyInjector.injectCaFrom }} + {{- end }} + {{- if .Values.proxyInjector.injectCaFromSecret }} + cert-manager.io/inject-ca-from-secret: {{ .Values.proxyInjector.injectCaFromSecret }} + {{- end }} + {{- end }} + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +webhooks: +- name: linkerd-proxy-injector.linkerd.io + namespaceSelector: + {{- toYaml .Values.proxyInjector.namespaceSelector | trim | nindent 4 }} + objectSelector: + {{- toYaml .Values.proxyInjector.objectSelector | trim | nindent 4 }} + clientConfig: + service: + name: linkerd-proxy-injector + namespace: {{ .Release.Namespace }} + path: "/" + {{- if and (empty .Values.proxyInjector.injectCaFrom) (empty .Values.proxyInjector.injectCaFromSecret) }} + caBundle: {{ ternary (b64enc (trim $ca.Cert)) (b64enc (trim .Values.proxyInjector.caBundle)) (empty .Values.proxyInjector.caBundle) }} + {{- end }} + failurePolicy: {{.Values.webhookFailurePolicy}} + admissionReviewVersions: ["v1", "v1beta1"] + rules: + - operations: [ "CREATE" ] + apiGroups: [""] + apiVersions: ["v1"] + resources: ["pods", "services"] + scope: "Namespaced" + sideEffects: None + timeoutSeconds: {{ .Values.proxyInjector.timeoutSeconds | default 10 }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector.yaml new file mode 100644 index 0000000000..7d514dbf06 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/proxy-injector.yaml @@ -0,0 +1,227 @@ +--- +### +### Proxy Injector +### +{{- $tree := deepCopy . }} +{{ $_ := set $tree.Values.proxy "workloadKind" "deployment" -}} +{{ $_ := set $tree.Values.proxy "component" "linkerd-proxy-injector" -}} +{{ $_ := set $tree.Values.proxy "waitBeforeExitSeconds" 0 -}} +{{- if not (empty .Values.proxyInjectorProxyResources) }} +{{- $c := dig "cores" .Values.proxy.cores .Values.proxyInjectorProxyResources }} +{{- $_ := set $tree.Values.proxy "cores" $c }} +{{- $r := merge .Values.proxyInjectorProxyResources .Values.proxy.resources }} +{{- $_ := set $tree.Values.proxy "resources" $r }} +{{- end }} +apiVersion: apps/v1 +kind: Deployment +metadata: + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + app.kubernetes.io/name: proxy-injector + app.kubernetes.io/part-of: Linkerd + app.kubernetes.io/version: {{.Values.linkerdVersion}} + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + name: linkerd-proxy-injector + namespace: {{ .Release.Namespace }} +spec: + replicas: {{.Values.controllerReplicas}} + revisionHistoryLimit: {{.Values.revisionHistoryLimit}} + selector: + matchLabels: + linkerd.io/control-plane-component: proxy-injector + {{- if .Values.deploymentStrategy }} + strategy: + {{- with .Values.deploymentStrategy }}{{ toYaml . | trim | nindent 4 }}{{- end }} + {{- end }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/proxy-injector-rbac.yaml") . | sha256sum }} + {{ include "partials.annotations.created-by" . }} + {{- include "partials.proxy.annotations" . | nindent 8}} + {{- with .Values.podAnnotations }}{{ toYaml . | trim | nindent 8 }}{{- end }} + config.linkerd.io/opaque-ports: "8443" + config.linkerd.io/default-inbound-policy: "all-unauthenticated" + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + linkerd.io/workload-ns: {{.Release.Namespace}} + {{- include "partials.proxy.labels" $tree.Values.proxy | nindent 8}} + {{- with .Values.podLabels }}{{ toYaml . | trim | nindent 8 }}{{- end }} + spec: + {{- with .Values.runtimeClassName }} + runtimeClassName: {{ . | quote }} + {{- end }} + {{- if .Values.tolerations -}} + {{- include "linkerd.tolerations" . | nindent 6 }} + {{- end -}} + {{- include "linkerd.node-selector" . | nindent 6 }} + {{- $_ := set $tree "component" "proxy-injector" -}} + {{- include "linkerd.affinity" $tree | nindent 6 }} + automountServiceAccountToken: false + containers: + {{- $_ := set $tree.Values.proxy "await" $tree.Values.proxy.await }} + {{- $_ := set $tree.Values.proxy "loadTrustBundleFromConfigMap" true }} + {{- $_ := set $tree.Values.proxy "podInboundPorts" "8443,9995" }} + {{- /* + The pod needs to accept webhook traffic, and we can't rely on that originating in the + cluster network. + */}} + {{- $_ := set $tree.Values.proxy "defaultInboundPolicy" "all-unauthenticated" }} + {{- $_ := set $tree.Values.proxy "capabilities" (dict "drop" (list "ALL")) }} + {{- $_ := set $tree.Values.proxy "outboundDiscoveryCacheUnusedTimeout" "5s" }} + {{- $_ := set $tree.Values.proxy "inboundDiscoveryCacheUnusedTimeout" "90s" }} + {{- if not $tree.Values.proxy.nativeSidecar }} + - {{- include "partials.proxy" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{- end }} + - args: + - proxy-injector + - -log-level={{.Values.controllerLogLevel}} + - -log-format={{.Values.controllerLogFormat}} + - -linkerd-namespace={{.Release.Namespace}} + - -enable-pprof={{.Values.enablePprof | default false}} + {{- if or (.Values.proxyInjector).additionalEnv (.Values.proxyInjector).experimentalEnv }} + env: + {{- with (.Values.proxyInjector).additionalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- with (.Values.proxyInjector).experimentalEnv }} + {{- toYaml . | nindent 8 -}} + {{- end }} + {{- end }} + image: {{.Values.controllerImage}}:{{.Values.controllerImageVersion | default .Values.linkerdVersion}} + imagePullPolicy: {{.Values.imagePullPolicy}} + livenessProbe: + httpGet: + path: /ping + port: 9995 + initialDelaySeconds: 10 + {{- with (.Values.proxyInjector.livenessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + name: proxy-injector + ports: + - containerPort: 8443 + name: proxy-injector + - containerPort: 9995 + name: admin-http + readinessProbe: + failureThreshold: 7 + httpGet: + path: /ready + port: 9995 + {{- with (.Values.proxyInjector.readinessProbe).timeoutSeconds }} + timeoutSeconds: {{ . }} + {{- end }} + {{- if .Values.proxyInjectorResources -}} + {{- include "partials.resources" .Values.proxyInjectorResources | nindent 8 }} + {{- end }} + securityContext: + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.controllerUID}} + {{- if ge (int .Values.controllerGID) 0 }} + runAsGroup: {{.Values.controllerGID}} + {{- end }} + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /var/run/linkerd/config + name: config + - mountPath: /var/run/linkerd/identity/trust-roots + name: trust-roots + - mountPath: /var/run/linkerd/tls + name: tls + readOnly: true + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: kube-api-access + readOnly: true + initContainers: + {{ if .Values.cniEnabled -}} + - {{- include "partials.network-validator" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ else -}} + {{- /* + The controller needs to connect to the Kubernetes API. There's no reason + to put the proxy in the way of that. + */}} + {{- $_ := set $tree.Values.proxyInit "ignoreOutboundPorts" .Values.proxyInit.kubeAPIServerPorts -}} + - {{- include "partials.proxy-init" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{- if $tree.Values.proxy.nativeSidecar }} + {{- $_ := set $tree.Values.proxy "startupProbeInitialDelaySeconds" 35 }} + {{- $_ := set $tree.Values.proxy "startupProbePeriodSeconds" 5 }} + {{- $_ := set $tree.Values.proxy "startupProbeFailureThreshold" 20 }} + - {{- include "partials.proxy" $tree | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{- if .Values.priorityClassName -}} + priorityClassName: {{ .Values.priorityClassName }} + {{ end -}} + securityContext: + seccompProfile: + type: RuntimeDefault + serviceAccountName: linkerd-proxy-injector + volumes: + - configMap: + name: linkerd-config + name: config + - configMap: + name: linkerd-identity-trust-roots + name: trust-roots + - name: tls + secret: + secretName: linkerd-proxy-injector-k8s-tls + - {{- include "partials.volumes.manual-mount-service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ if not .Values.cniEnabled -}} + - {{- include "partials.proxyInit.volumes.xtables" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + {{if .Values.identity.serviceAccountTokenProjection -}} + - {{- include "partials.proxy.volumes.service-account-token" . | indent 8 | trimPrefix (repeat 7 " ") }} + {{ end -}} + - {{- include "partials.proxy.volumes.identity" . | indent 8 | trimPrefix (repeat 7 " ") }} +--- +kind: Service +apiVersion: v1 +metadata: + name: linkerd-proxy-injector + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} + config.linkerd.io/opaque-ports: "443" +spec: + type: ClusterIP + selector: + linkerd.io/control-plane-component: proxy-injector + ports: + - name: proxy-injector + port: 443 + targetPort: proxy-injector +{{- if .Values.enablePodDisruptionBudget }} +--- +kind: PodDisruptionBudget +apiVersion: policy/v1 +metadata: + name: linkerd-proxy-injector + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-component: proxy-injector + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} + annotations: + {{ include "partials.annotations.created-by" . }} +spec: + maxUnavailable: {{ .Values.controller.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + linkerd.io/control-plane-component: proxy-injector +{{- end }} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/templates/psp.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/psp.yaml new file mode 100644 index 0000000000..db91fea675 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/templates/psp.yaml @@ -0,0 +1,119 @@ +{{ if .Values.enablePSP -}} +--- +### +### Control Plane PSP +### +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: linkerd-{{.Release.Namespace}}-control-plane + annotations: + seccomp.security.alpha.kubernetes.io/allowedProfileNames: "runtime/default" + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +spec: + {{- if or .Values.proxyInit.closeWaitTimeoutSecs .Values.proxyInit.runAsRoot }} + allowPrivilegeEscalation: true + {{- else }} + allowPrivilegeEscalation: false + {{- end }} + readOnlyRootFilesystem: true + {{- if empty .Values.cniEnabled }} + allowedCapabilities: + - NET_ADMIN + - NET_RAW + {{- end}} + requiredDropCapabilities: + - ALL + hostNetwork: false + hostIPC: false + hostPID: false + seLinux: + rule: RunAsAny + runAsUser: + {{- if .Values.cniEnabled }} + rule: MustRunAsNonRoot + {{- else }} + rule: RunAsAny + {{- end }} + runAsGroup: + {{- if .Values.cniEnabled }} + rule: MustRunAs + ranges: + - min: 1000 + max: 999999 + {{- else }} + rule: RunAsAny + {{- end }} + supplementalGroups: + rule: MustRunAs + ranges: + {{- if .Values.cniEnabled }} + - min: 10001 + max: 65535 + {{- else }} + - min: 1 + max: 65535 + {{- end }} + fsGroup: + rule: MustRunAs + ranges: + {{- if .Values.cniEnabled }} + - min: 10001 + max: 65535 + {{- else }} + - min: 1 + max: 65535 + {{- end }} + volumes: + - configMap + - emptyDir + - secret + - projected + - downwardAPI + - persistentVolumeClaim +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: linkerd-psp + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +rules: +- apiGroups: ['policy', 'extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - linkerd-{{.Release.Namespace}}-control-plane +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: linkerd-psp + namespace: {{ .Release.Namespace }} + labels: + linkerd.io/control-plane-ns: {{.Release.Namespace}} + {{- with .Values.commonLabels }}{{ toYaml . | trim | nindent 4 }}{{- end }} +roleRef: + kind: Role + name: linkerd-psp + apiGroup: rbac.authorization.k8s.io +subjects: +- kind: ServiceAccount + name: linkerd-destination + namespace: {{.Release.Namespace}} +{{ if not .Values.disableHeartBeat -}} +- kind: ServiceAccount + name: linkerd-heartbeat + namespace: {{.Release.Namespace}} +{{ end -}} +- kind: ServiceAccount + name: linkerd-identity + namespace: {{.Release.Namespace}} +- kind: ServiceAccount + name: linkerd-proxy-injector + namespace: {{.Release.Namespace}} +{{ end -}} diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/values-ha.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/values-ha.yaml new file mode 100644 index 0000000000..e3b8cbc070 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/values-ha.yaml @@ -0,0 +1,63 @@ +# This values.yaml file contains the values needed to enable HA mode. +# Usage: +# helm install -f values-ha.yaml + +# -- Create PodDisruptionBudget resources for each control plane workload +enablePodDisruptionBudget: true + +controller: + # -- sets pod disruption budget parameter for all deployments + podDisruptionBudget: + # -- Maximum number of pods that can be unavailable during disruption + maxUnavailable: 1 + +# -- Specify a deployment strategy for each control plane workload +deploymentStrategy: + rollingUpdate: + maxUnavailable: 1 + maxSurge: 25% + +# -- add PodAntiAffinity to each control plane workload +enablePodAntiAffinity: true + +# nodeAffinity: + +# proxy configuration +proxy: + resources: + cpu: + request: 100m + memory: + limit: 250Mi + request: 20Mi + +# controller configuration +controllerReplicas: 3 +controllerResources: &controller_resources + cpu: &controller_resources_cpu + limit: "" + request: 100m + memory: + limit: 250Mi + request: 50Mi +destinationResources: *controller_resources + +# identity configuration +identityResources: + cpu: *controller_resources_cpu + memory: + limit: 250Mi + request: 10Mi + +# heartbeat configuration +heartbeatResources: *controller_resources + +# proxy injector configuration +proxyInjectorResources: *controller_resources +webhookFailurePolicy: Fail + +# service profile validator configuration +spValidatorResources: *controller_resources + +# flag for linkerd check +highAvailability: true diff --git a/charts/buoyant/linkerd-control-plane/2024.11.3/values.yaml b/charts/buoyant/linkerd-control-plane/2024.11.3/values.yaml new file mode 100644 index 0000000000..39216eaa29 --- /dev/null +++ b/charts/buoyant/linkerd-control-plane/2024.11.3/values.yaml @@ -0,0 +1,671 @@ +# Default values for linkerd. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# -- Kubernetes DNS Domain name to use +clusterDomain: cluster.local + +# -- The cluster networks for which service discovery is performed. This should +# include the pod and service networks, but need not include the node network. +# +# By default, all IPv4 private networks and all accepted IPv6 ULAs are +# specified so that resolution works in typical Kubernetes environments. +clusterNetworks: "10.0.0.0/8,100.64.0.0/10,172.16.0.0/12,192.168.0.0/16,fd00::/8" +# -- Docker image pull policy +imagePullPolicy: IfNotPresent +# -- Specifies the number of old ReplicaSets to retain to allow rollback. +revisionHistoryLimit: 10 +# -- Log level for the control plane components +controllerLogLevel: info +# -- Log format for the control plane components +controllerLogFormat: plain +# -- enables control plane tracing +controlPlaneTracing: false +# -- namespace to send control plane traces to +controlPlaneTracingNamespace: linkerd-jaeger +# -- control plane version. See Proxy section for proxy version +linkerdVersion: edge-24.11.3 +# -- default kubernetes deployment strategy +deploymentStrategy: + rollingUpdate: + maxUnavailable: 25% + maxSurge: 25% +# -- enables the use of EndpointSlice informers for the destination service; +# enableEndpointSlices should be set to true only if EndpointSlice K8s feature +# gate is on +enableEndpointSlices: true +# -- enables pod anti affinity creation on deployments for high availability +enablePodAntiAffinity: false +# -- enables the use of pprof endpoints on control plane component's admin +# servers +enablePprof: false +# -- enables the creation of pod disruption budgets for control plane components +enablePodDisruptionBudget: false +# -- disables routing IPv6 traffic in addition to IPv4 traffic through the +# proxy (IPv6 routing only available as of proxy-init v2.3.0 and linkerd-cni +# v1.4.0) +disableIPv6: true + +controller: + # -- sets pod disruption budget parameter for all deployments + podDisruptionBudget: + # -- Maximum number of pods that can be unavailable during disruption + maxUnavailable: 1 +# -- enabling this omits the NET_ADMIN capability in the PSP +# and the proxy-init container when injecting the proxy; +# requires the linkerd-cni plugin to already be installed +cniEnabled: false +# -- Trust root certificate (ECDSA). It must be provided during install. +identityTrustAnchorsPEM: | +# -- Trust domain used for identity +# @default -- clusterDomain +identityTrustDomain: "" +kubeAPI: &kubeapi + # -- Maximum QPS sent to the kube-apiserver before throttling. + # See [token bucket rate limiter + # implementation](https://github.com/kubernetes/client-go/blob/v12.0.0/util/flowcontrol/throttle.go) + clientQPS: 100 + # -- Burst value over clientQPS + clientBurst: 200 +# -- Additional annotations to add to all pods +podAnnotations: {} +# -- Additional labels to add to all pods +podLabels: {} +# -- Labels to apply to all resources +commonLabels: {} +# -- Kubernetes priorityClassName for the Linkerd Pods +priorityClassName: "" +# -- Runtime Class Name for all the pods +runtimeClassName: "" + +# policy controller configuration +policyController: + image: + # -- Docker image for the policy controller + name: cr.l5d.io/linkerd/policy-controller + # -- Pull policy for the policy controller container image + # @default -- imagePullPolicy + pullPolicy: "" + # -- Tag for the policy controller container image + # @default -- linkerdVersion + version: "" + + # -- Log level for the policy controller + logLevel: info + + # -- The networks from which probes are performed. + # + # By default, all networks are allowed so that all probes are authorized. + probeNetworks: + - 0.0.0.0/0 + - "::/0" + + # -- policy controller resource requests & limits + resources: + cpu: + # -- Maximum amount of CPU units that the policy controller can use + limit: "" + # -- Amount of CPU units that the policy controller requests + request: "" + memory: + # -- Maximum amount of memory that the policy controller can use + limit: "" + # -- Maximum amount of memory that the policy controller requests + request: "" + ephemeral-storage: + # -- Maximum amount of ephemeral storage that the policy controller can use + limit: "" + # -- Amount of ephemeral storage that the policy controller requests + request: "" + + livenessProbe: + timeoutSeconds: 1 + readinessProbe: + timeoutSeconds: 1 + +# proxy configuration +proxy: + # -- Enable service profiles for non-Kubernetes services + enableExternalProfiles: false + # -- Maximum time allowed for the proxy to establish an outbound TCP + # connection + outboundConnectTimeout: 1000ms + # -- Maximum time allowed for the proxy to establish an inbound TCP + # connection + inboundConnectTimeout: 100ms + # -- Maximum time allowed before an unused outbound discovery result + # is evicted from the cache + outboundDiscoveryCacheUnusedTimeout: "5s" + # -- Maximum time allowed before an unused inbound discovery result + # is evicted from the cache + inboundDiscoveryCacheUnusedTimeout: "90s" + # -- When set to true, disables the protocol detection timeout on the + # outbound side of the proxy by setting it to a very high value + disableOutboundProtocolDetectTimeout: false + # -- When set to true, disables the protocol detection timeout on the inbound + # side of the proxy by setting it to a very high value + disableInboundProtocolDetectTimeout: false + image: + # -- Docker image for the proxy + name: cr.l5d.io/linkerd/proxy + # -- Pull policy for the proxy container image + # @default -- imagePullPolicy + pullPolicy: "" + # -- Tag for the proxy container image + # @default -- linkerdVersion + version: "" + # -- Enables the proxy's /shutdown admin endpoint + enableShutdownEndpoint: false + # -- Log level for the proxy + logLevel: warn,linkerd=info,hickory=error + # -- Log format (`plain` or `json`) for the proxy + logFormat: plain + # -- (`off` or `insecure`) If set to `off`, will prevent the proxy from + # logging HTTP headers. If set to `insecure`, HTTP headers may be logged + # verbatim. Note that setting this to `insecure` is not alone sufficient to + # log HTTP headers; the proxy logLevel must also be set to debug. + logHTTPHeaders: "off" + ports: + # -- Admin port for the proxy container + admin: 4191 + # -- Control port for the proxy container + control: 4190 + # -- Inbound port for the proxy container + inbound: 4143 + # -- Outbound port for the proxy container + outbound: 4140 + # -- The `cpu.limit` and `cores` should be kept in sync. The value of `cores` + # must be an integer and should typically be set by rounding up from the + # limit. E.g. if cpu.limit is '1500m', cores should be 2. + cores: 0 + resources: + cpu: + # -- Maximum amount of CPU units that the proxy can use + limit: "" + # -- Amount of CPU units that the proxy requests + request: "" + memory: + # -- Maximum amount of memory that the proxy can use + limit: "" + # -- Maximum amount of memory that the proxy requests + request: "" + ephemeral-storage: + # -- Maximum amount of ephemeral storage that the proxy can use + limit: "" + # -- Amount of ephemeral storage that the proxy requests + request: "" + # -- User id under which the proxy runs + uid: 2102 + # -- (int) Optional customisation of the group id under which the proxy runs (the group ID will be omitted if lower than 0) + gid: -1 + + # -- If set the injected proxy sidecars in the data plane will stay alive for + # at least the given period before receiving the SIGTERM signal from + # Kubernetes but no longer than the pod's `terminationGracePeriodSeconds`. + # See [Lifecycle + # hooks](https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks) + # for more info on container lifecycle hooks. + waitBeforeExitSeconds: 0 + # -- If set, the application container will not start until the proxy is + # ready + await: true + requireIdentityOnInboundPorts: "" + # -- Default set of opaque ports + # - SMTP (25,587) server-first + # - MYSQL (3306) server-first + # - Galera (4444) server-first + # - PostgreSQL (5432) server-first + # - Redis (6379) server-first + # - ElasticSearch (9300) server-first + # - Memcached (11211) clients do not issue any preamble, which breaks detection + opaquePorts: "25,587,3306,4444,5432,6379,9300,11211" + # -- Grace period for graceful proxy shutdowns. If this timeout elapses before all open connections have completed, the proxy will terminate forcefully, closing any remaining connections. + shutdownGracePeriod: "" + # -- The default allow policy to use when no `Server` selects a pod. One of: "all-authenticated", + # "all-unauthenticated", "cluster-authenticated", "cluster-unauthenticated", "deny", "audit" + # @default -- "all-unauthenticated" + defaultInboundPolicy: "all-unauthenticated" + # -- Enable KEP-753 native sidecars + # This is an experimental feature. It requires Kubernetes >= 1.29. + # If enabled, .proxy.waitBeforeExitSeconds should not be used. + nativeSidecar: false + # -- Native sidecar proxy startup probe parameters. + # -- LivenessProbe timeout and delay configuration + livenessProbe: + initialDelaySeconds: 10 + timeoutSeconds: 1 + # -- ReadinessProbe timeout and delay configuration + readinessProbe: + initialDelaySeconds: 2 + timeoutSeconds: 1 + startupProbe: + initialDelaySeconds: 0 + periodSeconds: 1 + failureThreshold: 120 + # Configures general properties of the proxy's control plane clients. + control: + # Configures limits on API response streams. + streams: + # -- The timeout for the first update from the control plane. + initialTimeout: "3s" + # -- The timeout between consecutive updates from the control plane. + idleTimeout: "5m" + # -- The maximum duration for a response stream (i.e. before it will be + # reinitialized). + lifetime: "1h" + inbound: + server: + http2: + # -- The interval at which PINGs are issued to remote HTTP/2 clients. + keepAliveInterval: "10s" + # -- The timeout within which keep-alive PINGs must be acknowledged on inbound HTTP/2 connections. + keepAliveTimeout: "3s" + outbound: + server: + http2: + # -- The interval at which PINGs are issued to local application HTTP/2 clients. + keepAliveInterval: "10s" + # -- The timeout within which keep-alive PINGs must be acknowledged on outbound HTTP/2 connections. + keepAliveTimeout: "3s" + +# proxy-init configuration +proxyInit: + # -- Variant of iptables that will be used to configure routing. Currently, + # proxy-init can be run either in 'nft' or in 'legacy' mode. The mode will + # control which utility binary will be called. The host must support + # whichever mode will be used + iptablesMode: "legacy" + # -- Default set of inbound ports to skip via iptables + # - Galera (4567,4568) + ignoreInboundPorts: "4567,4568" + # -- Default set of outbound ports to skip via iptables + # - Galera (4567,4568) + ignoreOutboundPorts: "4567,4568" + # -- Default set of ports to skip via iptables for control plane + # components so they can communicate with the Kubernetes API Server + kubeAPIServerPorts: "443,6443" + # -- Comma-separated list of subnets in valid CIDR format that should be skipped by the proxy + skipSubnets: "" + # -- Log level for the proxy-init + # @default -- info + logLevel: "" + # -- Log format (`plain` or `json`) for the proxy-init + # @default -- plain + logFormat: "" + image: + # -- Docker image for the proxy-init container + name: cr.l5d.io/linkerd/proxy-init + # -- Pull policy for the proxy-init container image + # @default -- imagePullPolicy + pullPolicy: "" + # -- Tag for the proxy-init container image + version: v2.4.1 + closeWaitTimeoutSecs: 0 + # -- Privileged mode allows the container processes to inherit all security + # capabilities and bypass any security limitations enforced by the kubelet. + # When used with 'runAsRoot: true', the container will behave exactly as if + # it was running as root on the host. May escape cgroup limits and see other + # processes and devices on the host. + # @default -- false + privileged: false + # -- Allow overriding the runAsNonRoot behaviour () + runAsRoot: false + # -- This value is used only if runAsRoot is false; otherwise runAsUser will be 0 + runAsUser: 65534 + # -- This value is used only if runAsRoot is false; otherwise runAsGroup will be 0 + runAsGroup: 65534 + xtMountPath: + mountPath: /run + name: linkerd-proxy-init-xtables-lock + +# network validator configuration +# This runs on a host that uses iptables to reroute network traffic. The validator +# ensures that iptables is correctly routing requests before we start linkerd. +networkValidator: + # -- Log level for the network-validator + # @default -- debug + logLevel: debug + # -- Log format (`plain` or `json`) for network-validator + # @default -- plain + logFormat: plain + # -- Address to which the network-validator will attempt to connect. This should be an IP + # that the cluster is expected to be able to reach but a port it should not, e.g., a public IP + # for public clusters and a private IP for air-gapped clusters with a port like 20001. + # If empty, defaults to 1.1.1.1:20001 and [fd00::1]:20001 for IPv4 and IPv6 respectively. + connectAddr: "" + # -- Address to which network-validator listens to requests from itself. + # If empty, defaults to 0.0.0.0:4140 and [::]:4140 for IPv4 and IPv6 respectively. + listenAddr: "" + # -- Timeout before network-validator fails to validate the pod's network connectivity + timeout: "10s" + # -- Include a securityContext in the network-validator pod spec + enableSecurityContext: true + +# -- For Private docker registries, authentication is needed. +# Registry secrets are applied to the respective service accounts +imagePullSecrets: [] +# - name: my-private-docker-registry-login-secret + +# -- Allow proxies to perform transparent HTTP/2 upgrading +enableH2Upgrade: true + +# -- Add a PSP resource and bind it to the control plane ServiceAccounts. Note +# PSP has been deprecated since k8s v1.21 +enablePSP: false + +# -- Failure policy for the proxy injector +webhookFailurePolicy: Ignore + +# controllerImage -- Docker image for the destination and identity components +controllerImage: cr.l5d.io/linkerd/controller +# -- Optionally allow a specific container image Tag (or SHA) to be specified for the controllerImage. +controllerImageVersion: "" + +# -- Number of replicas for each control plane pod +controllerReplicas: 1 +# -- User ID for the control plane components +controllerUID: 2103 +# -- (int) Optional customisation of the group ID for the control plane components (the group ID will be omitted if lower than 0) +controllerGID: -1 + +# destination configuration +# set resources for the sp-validator and its linkerd proxy respectively +# see proxy.resources for details. +# destinationResources -- CPU, Memory and Ephemeral Storage resources required by destination (see `proxy.resources` for sub-fields) +#destinationResources: +# destinationProxyResources -- CPU, Memory and Ephemeral Storage resources required by proxy injected into destination pod (see `proxy.resources` for sub-fields) +#destinationProxyResources: + +destinationController: + meshedHttp2ClientProtobuf: + keep_alive: + interval: + seconds: 10 + timeout: + seconds: 3 + while_idle: true + livenessProbe: + timeoutSeconds: 1 + readinessProbe: + timeoutSeconds: 1 + +# debug configuration +debugContainer: + image: + # -- Docker image for the debug container + name: cr.l5d.io/linkerd/debug + # -- Pull policy for the debug container image + # @default -- imagePullPolicy + pullPolicy: "" + # -- Tag for the debug container image + # @default -- linkerdVersion + version: "" + +identity: + # -- If the linkerd-identity-trust-roots ConfigMap has already been created + externalCA: false + + # -- Use [Service Account token Volume projection](https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#service-account-token-volume-projection) for pod validation instead of the default token + serviceAccountTokenProjection: true + + issuer: + scheme: linkerd.io/tls + + # -- Amount of time to allow for clock skew within a Linkerd cluster + clockSkewAllowance: 20s + + # -- Amount of time for which the Identity issuer should certify identity + issuanceLifetime: 24h0m0s + + # -- Which scheme is used for the identity issuer secret format + tls: + # -- Issuer certificate (ECDSA). It must be provided during install. + crtPEM: | + + # -- Key for the issuer certificate (ECDSA). It must be provided during + # install + keyPEM: | + + kubeAPI: *kubeapi + + livenessProbe: + timeoutSeconds: 1 + readinessProbe: + timeoutSeconds: 1 + +# -|- CPU, Memory and Ephemeral Storage resources required by the identity controller (see `proxy.resources` for sub-fields) +#identityResources: +# -|- CPU, Memory and Ephemeral Storage resources required by proxy injected into identity pod (see `proxy.resources` for sub-fields) +#identityProxyResources: + +# heartbeat configuration +# disableHeartBeat -- Set to true to not start the heartbeat cronjob +disableHeartBeat: false +# -- Config for the heartbeat cronjob +# heartbeatSchedule: "0 0 * * *" + +# proxy injector configuration +proxyInjector: + # -- Timeout in seconds before the API Server cancels a request to the proxy + # injector. If timeout is exceeded, the webhookfailurePolicy is used. + timeoutSeconds: 10 + # -- Do not create a secret resource for the proxyInjector webhook. + # If this is set to `true`, the value `proxyInjector.caBundle` must be set + # or the ca bundle must injected with cert-manager ca injector using + # `proxyInjector.injectCaFrom` or `proxyInjector.injectCaFromSecret` (see below). + externalSecret: false + + # -- Namespace selector used by admission webhook. + namespaceSelector: + matchExpressions: + - key: config.linkerd.io/admission-webhooks + operator: NotIn + values: + - disabled + - key: kubernetes.io/metadata.name + operator: NotIn + values: + - kube-system + - cert-manager + + # -- Object selector used by admission webhook. + objectSelector: + matchExpressions: + - key: linkerd.io/control-plane-component + operator: DoesNotExist + - key: linkerd.io/cni-resource + operator: DoesNotExist + + # -- Certificate for the proxy injector. If not provided and not using an external secret + # then Helm will generate one. + crtPEM: | + + # -- Certificate key for the proxy injector. If not provided and not using an external secret + # then Helm will generate one. + keyPEM: | + + # -- Bundle of CA certificates for proxy injector. + # If not provided nor injected with cert-manager, + # then Helm will use the certificate generated for `proxyInjector.crtPEM`. + # If `proxyInjector.externalSecret` is set to true, this value, injectCaFrom, or + # injectCaFromSecret must be set, as no certificate will be generated. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. + caBundle: | + + # -- Inject the CA bundle from a cert-manager Certificate. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) + # for more information. + injectCaFrom: "" + + # -- Inject the CA bundle from a Secret. + # If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. + # The Secret must have the CA Bundle stored in the `ca.crt` key and have + # the `cert-manager.io/allow-direct-injection` annotation set to `true`. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) + # for more information. + injectCaFromSecret: "" + + livenessProbe: + timeoutSeconds: 1 + readinessProbe: + timeoutSeconds: 1 + +# -|- CPU, Memory and Ephemeral Storage resources required by the proxy injector (see +#`proxy.resources` for sub-fields) +#proxyInjectorResources: +#-|- CPU, Memory and Ephemeral Storage resources required by proxy injected into the proxy injector +#pod (see `proxy.resources` for sub-fields) +#proxyInjectorProxyResources: + +# service profile validator configuration +profileValidator: + # -- Do not create a secret resource for the profileValidator webhook. + # If this is set to `true`, the value `proxyInjector.caBundle` must be set + # or the ca bundle must injected with cert-manager ca injector using + # `proxyInjector.injectCaFrom` or `proxyInjector.injectCaFromSecret` (see below). + externalSecret: false + + # -- Namespace selector used by admission webhook + namespaceSelector: + matchExpressions: + - key: config.linkerd.io/admission-webhooks + operator: NotIn + values: + - disabled + + # -- Certificate for the service profile validator. If not provided and not using an external secret + # then Helm will generate one. + crtPEM: | + + # -- Certificate key for the service profile validator. If not provided and not using an external secret + # then Helm will generate one. + keyPEM: | + + # -- Bundle of CA certificates for proxy injector. + # If not provided nor injected with cert-manager, + # then Helm will use the certificate generated for `profileValidator.crtPEM`. + # If `profileValidator.externalSecret` is set to true, this value, injectCaFrom, or + # injectCaFromSecret must be set, as no certificate will be generated. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. + caBundle: | + + # -- Inject the CA bundle from a cert-manager Certificate. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) + # for more information. + injectCaFrom: "" + + # -- Inject the CA bundle from a Secret. + # If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. + # The Secret must have the CA Bundle stored in the `ca.crt` key and have + # the `cert-manager.io/allow-direct-injection` annotation set to `true`. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) + # for more information. + injectCaFromSecret: "" + +# policy validator configuration +policyValidator: + # -- Do not create a secret resource for the policyValidator webhook. + # If this is set to `true`, the value `policyValidator.caBundle` must be set + # or the ca bundle must injected with cert-manager ca injector using + # `policyValidator.injectCaFrom` or `policyValidator.injectCaFromSecret` (see below). + externalSecret: false + + # -- Namespace selector used by admission webhook + namespaceSelector: + matchExpressions: + - key: config.linkerd.io/admission-webhooks + operator: NotIn + values: + - disabled + + # -- Certificate for the policy validator. If not provided and not using an external secret + # then Helm will generate one. + crtPEM: | + + # -- Certificate key for the policy validator. If not provided and not using an external secret + # then Helm will generate one. + keyPEM: | + + # -- Bundle of CA certificates for proxy injector. + # If not provided nor injected with cert-manager, + # then Helm will use the certificate generated for `policyValidator.crtPEM`. + # If `policyValidator.externalSecret` is set to true, this value, injectCaFrom, or + # injectCaFromSecret must be set, as no certificate will be generated. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector) for more information. + caBundle: | + + # -- Inject the CA bundle from a cert-manager Certificate. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-certificate-resource) + # for more information. + injectCaFrom: "" + + # -- Inject the CA bundle from a Secret. + # If set, the `cert-manager.io/inject-ca-from-secret` annotation will be added to the webhook. + # The Secret must have the CA Bundle stored in the `ca.crt` key and have + # the `cert-manager.io/allow-direct-injection` annotation set to `true`. + # See the cert-manager [CA Injector Docs](https://cert-manager.io/docs/concepts/ca-injector/#injecting-ca-data-from-a-secret-resource) + # for more information. + injectCaFromSecret: "" + +# -- NodeSelector section, See the [K8S +# documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector) +# for more information +nodeSelector: + kubernetes.io/os: linux + +# -- SP validator configuration +spValidator: + livenessProbe: + timeoutSeconds: 1 + readinessProbe: + timeoutSeconds: 1 + +# -|- CPU, Memory and Ephemeral Storage resources required by the SP validator (see +#`proxy.resources` for sub-fields) +#spValidatorResources: + +# -|- Tolerations section, See the +# [K8S documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/) +# for more information +#tolerations: + +# -|- NodeAffinity section, See the +# [K8S documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity) +# for more information +#nodeAffinity: + +# -- url of external prometheus instance (used for the heartbeat) +prometheusUrl: "" + +# Prometheus Operator PodMonitor configuration +podMonitor: + # -- Enables the creation of Prometheus Operator [PodMonitor](https://prometheus-operator.dev/docs/operator/api/#monitoring.coreos.com/v1.PodMonitor) + enabled: false + # -- Interval at which metrics should be scraped + scrapeInterval: 10s + # -- Iimeout after which the scrape is ended + scrapeTimeout: 10s + # -- Labels to apply to all pod Monitors + labels: {} + controller: + # -- Enables the creation of PodMonitor for the control-plane + enabled: true + # -- Selector to select which namespaces the Endpoints objects are discovered from + namespaceSelector: | + matchNames: + - {{ .Release.Namespace }} + - linkerd-viz + - linkerd-jaeger + serviceMirror: + # -- Enables the creation of PodMonitor for the Service Mirror component + enabled: true + proxy: + # -- Enables the creation of PodMonitor for the data-plane + enabled: true + + +# Egress related configuration +egress: + # -- The namespace that is used to store egress configuration that affects all client workloads in the cluster + globalEgressNetworkNamespace: linkerd-egress + \ No newline at end of file diff --git a/charts/buoyant/linkerd-crds/2024.11.3/.helmignore b/charts/buoyant/linkerd-crds/2024.11.3/.helmignore new file mode 100644 index 0000000000..79c90a8063 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +OWNERS +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/charts/buoyant/linkerd-crds/2024.11.3/Chart.lock b/charts/buoyant/linkerd-crds/2024.11.3/Chart.lock new file mode 100644 index 0000000000..a62a030631 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: partials + repository: file://../partials + version: 0.1.0 +digest: sha256:8e42f9c9d4a2dc883f17f94d6044c97518ced19ad0922f47b8760e47135369ba +generated: "2021-08-17T10:42:52.610449255-05:00" diff --git a/charts/buoyant/linkerd-crds/2024.11.3/Chart.yaml b/charts/buoyant/linkerd-crds/2024.11.3/Chart.yaml new file mode 100644 index 0000000000..ee99d99751 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/Chart.yaml @@ -0,0 +1,26 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Linkerd CRDs + catalog.cattle.io/kube-version: '>=1.22.0-0' + catalog.cattle.io/release-name: linkerd-crds +apiVersion: v2 +dependencies: +- name: partials + repository: file://../partials + version: 0.1.0 +description: 'Linkerd gives you observability, reliability, and security for your + microservices — with no code change required. ' +home: https://linkerd.io +icon: file://assets/icons/linkerd-crds.png +keywords: +- service-mesh +kubeVersion: '>=1.22.0-0' +maintainers: +- email: cncf-linkerd-dev@lists.cncf.io + name: Linkerd authors + url: https://linkerd.io/ +name: linkerd-crds +sources: +- https://github.com/linkerd/linkerd2/ +type: application +version: 2024.11.3 diff --git a/charts/buoyant/linkerd-crds/2024.11.3/README.md b/charts/buoyant/linkerd-crds/2024.11.3/README.md new file mode 100644 index 0000000000..f390cd0a84 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/README.md @@ -0,0 +1,73 @@ +# linkerd-crds + +Linkerd gives you observability, reliability, and security +for your microservices — with no code change required. + +![Version: 2024.11.3](https://img.shields.io/badge/Version-2024.11.3-informational?style=flat-square) +![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) + +**Homepage:** + +## Quickstart and documentation + +You can run Linkerd on any Kubernetes cluster in a matter of seconds. See the +[Linkerd Getting Started Guide][getting-started] for how. + +For more comprehensive documentation, start with the [Linkerd +docs][linkerd-docs]. + +## Adding Linkerd's Helm repository + +```bash +# To add the repo for Linkerd edge releases: +helm repo add linkerd https://helm.linkerd.io/edge +``` + +## Installing the linkerd-crds chart + +This installs the `linkerd-crds` chart, which only persists the CRDs that +Linkerd requires. + +After installing this chart, you need then to install the +`linkerd-control-plane` chart in the same namespace, which provides all the +linkerd core control components. + +```bash +helm install linkerd-crds -n linkerd --create-namespace linkerd/linkerd-crds +``` + +## Get involved + +* Check out Linkerd's source code at [GitHub][linkerd2]. +* Join Linkerd's [user mailing list][linkerd-users], [developer mailing + list][linkerd-dev], and [announcements mailing list][linkerd-announce]. +* Follow [@linkerd][twitter] on Twitter. +* Join the [Linkerd Slack][slack]. + +[getting-started]: https://linkerd.io/2/getting-started/ +[linkerd2]: https://github.com/linkerd/linkerd2 +[linkerd-announce]: https://lists.cncf.io/g/cncf-linkerd-announce +[linkerd-dev]: https://lists.cncf.io/g/cncf-linkerd-dev +[linkerd-docs]: https://linkerd.io/2/overview/ +[linkerd-users]: https://lists.cncf.io/g/cncf-linkerd-users +[slack]: http://slack.linkerd.io +[twitter]: https://twitter.com/linkerd + +## Requirements + +Kubernetes: `>=1.22.0-0` + +| Repository | Name | Version | +|------------|------|---------| +| file://../partials | partials | 0.1.0 | + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| enableHttpRoutes | bool | `true` | | +| enableTcpRoutes | bool | `true` | | +| enableTlsRoutes | bool | `true` | | + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.12.0](https://github.com/norwoodj/helm-docs/releases/v1.12.0) diff --git a/charts/buoyant/linkerd-crds/2024.11.3/README.md.gotmpl b/charts/buoyant/linkerd-crds/2024.11.3/README.md.gotmpl new file mode 100644 index 0000000000..88be739549 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/README.md.gotmpl @@ -0,0 +1,59 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +{{ template "chart.versionBadge" . }} +{{ template "chart.typeBadge" . }} +{{ template "chart.appVersionBadge" . }} + +{{ template "chart.homepageLine" . }} + +## Quickstart and documentation + +You can run Linkerd on any Kubernetes cluster in a matter of seconds. See the +[Linkerd Getting Started Guide][getting-started] for how. + +For more comprehensive documentation, start with the [Linkerd +docs][linkerd-docs]. + +## Adding Linkerd's Helm repository + +```bash +# To add the repo for Linkerd edge releases: +helm repo add linkerd https://helm.linkerd.io/edge +``` + +## Installing the linkerd-crds chart + +This installs the `linkerd-crds` chart, which only persists the CRDs that +Linkerd requires. + +After installing this chart, you need then to install the +`linkerd-control-plane` chart in the same namespace, which provides all the +linkerd core control components. + +```bash +helm install linkerd-crds -n linkerd --create-namespace linkerd/linkerd-crds +``` + +## Get involved + +* Check out Linkerd's source code at [GitHub][linkerd2]. +* Join Linkerd's [user mailing list][linkerd-users], [developer mailing + list][linkerd-dev], and [announcements mailing list][linkerd-announce]. +* Follow [@linkerd][twitter] on Twitter. +* Join the [Linkerd Slack][slack]. + +[getting-started]: https://linkerd.io/2/getting-started/ +[linkerd2]: https://github.com/linkerd/linkerd2 +[linkerd-announce]: https://lists.cncf.io/g/cncf-linkerd-announce +[linkerd-dev]: https://lists.cncf.io/g/cncf-linkerd-dev +[linkerd-docs]: https://linkerd.io/2/overview/ +[linkerd-users]: https://lists.cncf.io/g/cncf-linkerd-users +[slack]: http://slack.linkerd.io +[twitter]: https://twitter.com/linkerd + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +{{ template "helm-docs.versionFooter" . }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/app-readme.md b/charts/buoyant/linkerd-crds/2024.11.3/app-readme.md new file mode 100644 index 0000000000..59010a6b21 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/app-readme.md @@ -0,0 +1,9 @@ +# Linkerd 2 CRDs Chart + +Linkerd is an ultra light, ultra simple, ultra powerful service mesh. Linkerd +adds security, observability, and reliability to Kubernetes, without the +complexity. + +This particular Helm chart only installs Linkerd CRDs. + +Full documentation available at: https://linkerd.io/2/overview/ diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/.helmignore b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/.helmignore @@ -0,0 +1,21 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/Chart.yaml b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/Chart.yaml new file mode 100644 index 0000000000..23cfc167e3 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/Chart.yaml @@ -0,0 +1,5 @@ +apiVersion: v1 +description: 'A Helm chart containing Linkerd partial templates, depended by the ''linkerd'' + and ''patch'' charts. ' +name: partials +version: 0.1.0 diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md new file mode 100644 index 0000000000..10805c9b94 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md @@ -0,0 +1,9 @@ +# partials + +A Helm chart containing Linkerd partial templates, +depended by the 'linkerd' and 'patch' charts. + +![Version: 0.1.0](https://img.shields.io/badge/Version-0.1.0-informational?style=flat-square) + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.12.0](https://github.com/norwoodj/helm-docs/releases/v1.12.0) diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md.gotmpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md.gotmpl new file mode 100644 index 0000000000..37f5101061 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/README.md.gotmpl @@ -0,0 +1,14 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +{{ template "chart.versionBadge" . }} +{{ template "chart.typeBadge" . }} +{{ template "chart.appVersionBadge" . }} + +{{ template "chart.homepageLine" . }} + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +{{ template "helm-docs.versionFooter" . }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/NOTES.txt b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/NOTES.txt new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_affinity.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_affinity.tpl new file mode 100644 index 0000000000..5dde1da473 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_affinity.tpl @@ -0,0 +1,38 @@ +{{ define "linkerd.pod-affinity" -}} +podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchExpressions: + - key: {{ default "linkerd.io/control-plane-component" .label }} + operator: In + values: + - {{ .component }} + topologyKey: topology.kubernetes.io/zone + weight: 100 + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: {{ default "linkerd.io/control-plane-component" .label }} + operator: In + values: + - {{ .component }} + topologyKey: kubernetes.io/hostname +{{- end }} + +{{ define "linkerd.node-affinity" -}} +nodeAffinity: +{{- toYaml .Values.nodeAffinity | trim | nindent 2 }} +{{- end }} + +{{ define "linkerd.affinity" -}} +{{- if or .Values.enablePodAntiAffinity .Values.nodeAffinity -}} +affinity: +{{- end }} +{{- if .Values.enablePodAntiAffinity -}} +{{- include "linkerd.pod-affinity" . | nindent 2 }} +{{- end }} +{{- if .Values.nodeAffinity -}} +{{- include "linkerd.node-affinity" . | nindent 2 }} +{{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_capabilities.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_capabilities.tpl new file mode 100644 index 0000000000..a595d74c1f --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_capabilities.tpl @@ -0,0 +1,16 @@ +{{- define "partials.proxy.capabilities" -}} +capabilities: + {{- if .Values.proxy.capabilities.add }} + add: + {{- toYaml .Values.proxy.capabilities.add | trim | nindent 4 }} + {{- end }} + {{- if .Values.proxy.capabilities.drop }} + drop: + {{- toYaml .Values.proxy.capabilities.drop | trim | nindent 4 }} + {{- end }} +{{- end -}} + +{{- define "partials.proxy-init.capabilities.drop" -}} +drop: +{{ toYaml .Values.proxyInit.capabilities.drop | trim }} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_debug.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_debug.tpl new file mode 100644 index 0000000000..4df8cc77bc --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_debug.tpl @@ -0,0 +1,15 @@ +{{- define "partials.debug" -}} +image: {{.Values.debugContainer.image.name}}:{{.Values.debugContainer.image.version | default .Values.linkerdVersion}} +imagePullPolicy: {{.Values.debugContainer.image.pullPolicy | default .Values.imagePullPolicy}} +name: linkerd-debug +terminationMessagePolicy: FallbackToLogsOnError +# some environments require probes, so we provide some infallible ones +livenessProbe: + exec: + command: + - "true" +readinessProbe: + exec: + command: + - "true" +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_helpers.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_helpers.tpl new file mode 100644 index 0000000000..b6cdc34d08 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_helpers.tpl @@ -0,0 +1,14 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Splits a coma separated list into a list of string values. +For example "11,22,55,44" will become "11","22","55","44" +*/}} +{{- define "partials.splitStringList" -}} +{{- if gt (len (toString .)) 0 -}} +{{- $ports := toString . | splitList "," -}} +{{- $last := sub (len $ports) 1 -}} +{{- range $i,$port := $ports -}} +"{{$port}}"{{ternary "," "" (ne $i $last)}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_metadata.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_metadata.tpl new file mode 100644 index 0000000000..04d2f1beab --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_metadata.tpl @@ -0,0 +1,17 @@ +{{- define "partials.annotations.created-by" -}} +linkerd.io/created-by: {{ .Values.cliVersion | default (printf "linkerd/helm %s" ( (.Values.image).version | default .Values.linkerdVersion)) }} +{{- end -}} + +{{- define "partials.proxy.annotations" -}} +linkerd.io/proxy-version: {{.Values.proxy.image.version | default .Values.linkerdVersion}} +cluster-autoscaler.kubernetes.io/safe-to-evict: "true" +linkerd.io/trust-root-sha256: {{ .Values.identityTrustAnchorsPEM | sha256sum }} +{{- end -}} + +{{/* +To add labels to the control-plane components, instead update at individual component manifests as +adding here would also update `spec.selector.matchLabels` which are immutable and would fail upgrades. +*/}} +{{- define "partials.proxy.labels" -}} +linkerd.io/proxy-{{.workloadKind}}: {{.component}} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_network-validator.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_network-validator.tpl new file mode 100644 index 0000000000..276056395f --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_network-validator.tpl @@ -0,0 +1,45 @@ +{{- define "partials.network-validator" -}} +name: linkerd-network-validator +image: {{.Values.proxy.image.name}}:{{.Values.proxy.image.version | default .Values.linkerdVersion }} +imagePullPolicy: {{.Values.proxy.image.pullPolicy | default .Values.imagePullPolicy}} +{{ include "partials.resources" .Values.proxy.resources }} +{{- if or .Values.networkValidator.enableSecurityContext }} +securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + readOnlyRootFilesystem: true + runAsGroup: 65534 + runAsNonRoot: true + runAsUser: 65534 + seccompProfile: + type: RuntimeDefault +{{- end }} +command: + - /usr/lib/linkerd/linkerd2-network-validator +args: + - --log-format + - {{ .Values.networkValidator.logFormat }} + - --log-level + - {{ .Values.networkValidator.logLevel }} + - --connect-addr + {{- if .Values.networkValidator.connectAddr }} + - {{ .Values.networkValidator.connectAddr | quote }} + {{- else if .Values.disableIPv6}} + - "1.1.1.1:20001" + {{- else }} + - "[fd00::1]:20001" + {{- end }} + - --listen-addr + {{- if .Values.networkValidator.listenAddr }} + - {{ .Values.networkValidator.listenAddr | quote }} + {{- else if .Values.disableIPv6}} + - "0.0.0.0:4140" + {{- else }} + - "[::]:4140" + {{- end }} + - --timeout + - {{ .Values.networkValidator.timeout }} + +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_nodeselector.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_nodeselector.tpl new file mode 100644 index 0000000000..4cde0ab16e --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_nodeselector.tpl @@ -0,0 +1,4 @@ +{{- define "linkerd.node-selector" -}} +nodeSelector: +{{- toYaml .Values.nodeSelector | trim | nindent 2 }} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl new file mode 100644 index 0000000000..9651b3bd1a --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-config-ann.tpl @@ -0,0 +1,18 @@ +{{- define "partials.proxy.config.annotations" -}} +{{- with .cpu }} +{{- with .request -}} +config.linkerd.io/proxy-cpu-request: {{. | quote}} +{{end}} +{{- with .limit -}} +config.linkerd.io/proxy-cpu-limit: {{. | quote}} +{{- end}} +{{- end}} +{{- with .memory }} +{{- with .request }} +config.linkerd.io/proxy-memory-request: {{. | quote}} +{{end}} +{{- with .limit -}} +config.linkerd.io/proxy-memory-limit: {{. | quote}} +{{- end}} +{{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-init.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-init.tpl new file mode 100644 index 0000000000..a307b14073 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy-init.tpl @@ -0,0 +1,98 @@ +{{- define "partials.proxy-init" -}} +args: +{{- if (.Values.proxyInit.iptablesMode | default "legacy" | eq "nft") }} +- --firewall-bin-path +- "iptables-nft" +- --firewall-save-bin-path +- "iptables-nft-save" +{{- else if not (eq .Values.proxyInit.iptablesMode "legacy") }} +{{ fail (printf "Unsupported value \"%s\" for proxyInit.iptablesMode\nValid values: [\"nft\", \"legacy\"]" .Values.proxyInit.iptablesMode) }} +{{end -}} +{{- if .Values.disableIPv6 }} +- --ipv6=false +{{- end }} +- --incoming-proxy-port +- {{.Values.proxy.ports.inbound | quote}} +- --outgoing-proxy-port +- {{.Values.proxy.ports.outbound | quote}} +- --proxy-uid +- {{.Values.proxy.uid | quote}} +{{- if ge (int .Values.proxy.gid) 0 }} +- --proxy-gid +- {{.Values.proxy.gid | quote}} +{{- end }} +- --inbound-ports-to-ignore +- "{{.Values.proxy.ports.control}},{{.Values.proxy.ports.admin}}{{ternary (printf ",%s" (.Values.proxyInit.ignoreInboundPorts | toString)) "" (not (empty .Values.proxyInit.ignoreInboundPorts)) }}" +{{- if .Values.proxyInit.ignoreOutboundPorts }} +- --outbound-ports-to-ignore +- {{.Values.proxyInit.ignoreOutboundPorts | quote}} +{{- end }} +{{- if .Values.proxyInit.closeWaitTimeoutSecs }} +- --timeout-close-wait-secs +- {{ .Values.proxyInit.closeWaitTimeoutSecs | quote}} +{{- end }} +{{- if .Values.proxyInit.logFormat }} +- --log-format +- {{ .Values.proxyInit.logFormat }} +{{- end }} +{{- if .Values.proxyInit.logLevel }} +- --log-level +- {{ .Values.proxyInit.logLevel }} +{{- end }} +{{- if .Values.proxyInit.skipSubnets }} +- --subnets-to-ignore +- {{ .Values.proxyInit.skipSubnets | quote }} +{{- end }} +image: {{.Values.proxyInit.image.name}}:{{.Values.proxyInit.image.version}} +imagePullPolicy: {{.Values.proxyInit.image.pullPolicy | default .Values.imagePullPolicy}} +name: linkerd-init +{{ include "partials.resources" .Values.proxy.resources }} +securityContext: + {{- if or .Values.proxyInit.closeWaitTimeoutSecs .Values.proxyInit.privileged }} + allowPrivilegeEscalation: true + {{- else }} + allowPrivilegeEscalation: false + {{- end }} + capabilities: + add: + - NET_ADMIN + - NET_RAW + {{- if .Values.proxyInit.capabilities -}} + {{- if .Values.proxyInit.capabilities.add }} + {{- toYaml .Values.proxyInit.capabilities.add | trim | nindent 4 }} + {{- end }} + {{- if .Values.proxyInit.capabilities.drop -}} + {{- include "partials.proxy-init.capabilities.drop" . | nindent 4 -}} + {{- end }} + {{- end }} + {{- if or .Values.proxyInit.closeWaitTimeoutSecs .Values.proxyInit.privileged }} + privileged: true + {{- else }} + privileged: false + {{- end }} + {{- if .Values.proxyInit.runAsRoot }} + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + {{- else }} + runAsNonRoot: true + runAsUser: {{ .Values.proxyInit.runAsUser | int | eq 0 | ternary 65534 .Values.proxyInit.runAsUser }} + runAsGroup: {{ .Values.proxyInit.runAsGroup | int | eq 0 | ternary 65534 .Values.proxyInit.runAsGroup }} + {{- end }} + readOnlyRootFilesystem: true + seccompProfile: + type: RuntimeDefault +terminationMessagePolicy: FallbackToLogsOnError +{{- if or (not .Values.cniEnabled) .Values.proxyInit.saMountPath }} +volumeMounts: +{{- end -}} +{{- if not .Values.cniEnabled }} +- mountPath: {{.Values.proxyInit.xtMountPath.mountPath}} + name: {{.Values.proxyInit.xtMountPath.name}} +{{- end -}} +{{- if .Values.proxyInit.saMountPath }} +- mountPath: {{.Values.proxyInit.saMountPath.mountPath}} + name: {{.Values.proxyInit.saMountPath.name}} + readOnly: {{.Values.proxyInit.saMountPath.readOnly}} +{{- end -}} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy.tpl new file mode 100644 index 0000000000..4dcf12dee2 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_proxy.tpl @@ -0,0 +1,271 @@ +{{ define "partials.proxy" -}} +{{ if and .Values.proxy.nativeSidecar .Values.proxy.waitBeforeExitSeconds }} +{{ fail "proxy.nativeSidecar and waitBeforeExitSeconds cannot be used simultaneously" }} +{{- end }} +{{- if not (has .Values.proxy.logHTTPHeaders (list "insecure" "off" "")) }} +{{- fail "logHTTPHeaders must be one of: insecure | off" }} +{{- end }} +{{- $trustDomain := (.Values.identityTrustDomain | default .Values.clusterDomain) -}} +env: +- name: _pod_name + valueFrom: + fieldRef: + fieldPath: metadata.name +- name: _pod_ns + valueFrom: + fieldRef: + fieldPath: metadata.namespace +- name: _pod_nodeName + valueFrom: + fieldRef: + fieldPath: spec.nodeName +{{- if .Values.proxy.cores }} +- name: LINKERD2_PROXY_CORES + value: {{.Values.proxy.cores | quote}} +{{- end }} +{{ if .Values.proxy.requireIdentityOnInboundPorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_REQUIRE_IDENTITY + value: {{.Values.proxy.requireIdentityOnInboundPorts | quote}} +{{ end -}} +{{ if .Values.proxy.requireTLSOnInboundPorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_REQUIRE_TLS + value: {{.Values.proxy.requireTLSOnInboundPorts | quote}} +{{ end -}} +- name: LINKERD2_PROXY_SHUTDOWN_ENDPOINT_ENABLED + value: {{.Values.proxy.enableShutdownEndpoint | quote}} +- name: LINKERD2_PROXY_LOG + value: "{{.Values.proxy.logLevel}}{{ if not (eq .Values.proxy.logHTTPHeaders "insecure") }},[{headers}]=off,[{request}]=off{{ end }}" +- name: LINKERD2_PROXY_LOG_FORMAT + value: {{.Values.proxy.logFormat | quote}} +- name: LINKERD2_PROXY_DESTINATION_SVC_ADDR + value: {{ternary "localhost.:8086" (printf "linkerd-dst-headless.%s.svc.%s.:8086" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-destination")}} +- name: LINKERD2_PROXY_DESTINATION_PROFILE_NETWORKS + value: {{.Values.clusterNetworks | quote}} +- name: LINKERD2_PROXY_POLICY_SVC_ADDR + value: {{ternary "localhost.:8090" (printf "linkerd-policy.%s.svc.%s.:8090" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-destination")}} +- name: LINKERD2_PROXY_POLICY_WORKLOAD + value: | + {"ns":"$(_pod_ns)", "pod":"$(_pod_name)"} +- name: LINKERD2_PROXY_INBOUND_DEFAULT_POLICY + value: {{.Values.proxy.defaultInboundPolicy}} +- name: LINKERD2_PROXY_POLICY_CLUSTER_NETWORKS + value: {{.Values.clusterNetworks | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_INITIAL_TIMEOUT + value: {{((.Values.proxy.control).streams).initialTimeout | default "" | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_IDLE_TIMEOUT + value: {{((.Values.proxy.control).streams).idleTimeout | default "" | quote}} +- name: LINKERD2_PROXY_CONTROL_STREAM_LIFETIME + value: {{((.Values.proxy.control).streams).lifetime | default "" | quote}} +{{ if .Values.proxy.inboundConnectTimeout -}} +- name: LINKERD2_PROXY_INBOUND_CONNECT_TIMEOUT + value: {{.Values.proxy.inboundConnectTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.outboundConnectTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_TIMEOUT + value: {{.Values.proxy.outboundConnectTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.outboundDiscoveryCacheUnusedTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_DISCOVERY_IDLE_TIMEOUT + value: {{.Values.proxy.outboundDiscoveryCacheUnusedTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.inboundDiscoveryCacheUnusedTimeout -}} +- name: LINKERD2_PROXY_INBOUND_DISCOVERY_IDLE_TIMEOUT + value: {{.Values.proxy.inboundDiscoveryCacheUnusedTimeout | quote}} +{{ end -}} +{{ if .Values.proxy.disableOutboundProtocolDetectTimeout -}} +- name: LINKERD2_PROXY_OUTBOUND_DETECT_TIMEOUT + value: "365d" +{{ end -}} +{{ if .Values.proxy.disableInboundProtocolDetectTimeout -}} +- name: LINKERD2_PROXY_INBOUND_DETECT_TIMEOUT + value: "365d" +{{ end -}} +- name: LINKERD2_PROXY_CONTROL_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.control}}" +- name: LINKERD2_PROXY_ADMIN_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.admin}}" +{{- /* Deprecated, superseded by LINKERD2_PROXY_OUTBOUND_LISTEN_ADDRS since proxy's v2.228.0 (deployed since edge-24.4.5) */}} +- name: LINKERD2_PROXY_OUTBOUND_LISTEN_ADDR + value: "127.0.0.1:{{.Values.proxy.ports.outbound}}" +- name: LINKERD2_PROXY_OUTBOUND_LISTEN_ADDRS + value: "127.0.0.1:{{.Values.proxy.ports.outbound}}{{ if not .Values.disableIPv6}},[::1]:{{.Values.proxy.ports.outbound}}{{ end }}" +- name: LINKERD2_PROXY_INBOUND_LISTEN_ADDR + value: "{{ if .Values.disableIPv6 }}0.0.0.0{{ else }}[::]{{ end }}:{{.Values.proxy.ports.inbound}}" +- name: LINKERD2_PROXY_INBOUND_IPS + valueFrom: + fieldRef: + fieldPath: status.podIPs +- name: LINKERD2_PROXY_INBOUND_PORTS + value: {{ .Values.proxy.podInboundPorts | quote }} +{{ if .Values.proxy.isGateway -}} +- name: LINKERD2_PROXY_INBOUND_GATEWAY_SUFFIXES + value: {{printf "svc.%s." .Values.clusterDomain}} +{{ end -}} +{{ if .Values.proxy.isIngress -}} +- name: LINKERD2_PROXY_INGRESS_MODE + value: "true" +{{ end -}} +- name: LINKERD2_PROXY_DESTINATION_PROFILE_SUFFIXES + {{- $internalDomain := printf "svc.%s." .Values.clusterDomain }} + value: {{ternary "." $internalDomain .Values.proxy.enableExternalProfiles}} +- name: LINKERD2_PROXY_INBOUND_ACCEPT_KEEPALIVE + value: 10000ms +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_KEEPALIVE + value: 10000ms +- name: LINKERD2_PROXY_INBOUND_ACCEPT_USER_TIMEOUT + value: 30s +- name: LINKERD2_PROXY_OUTBOUND_CONNECT_USER_TIMEOUT + value: 30s +{{- /* Configure inbound and outbound parameters, e.g. for HTTP/2 servers. */}} +{{ range $proxyK, $proxyV := (dict "inbound" .Values.proxy.inbound "outbound" .Values.proxy.outbound) -}} +{{ range $scopeK, $scopeV := $proxyV -}} +{{ range $protoK, $protoV := $scopeV -}} +{{ range $paramK, $paramV := $protoV -}} +- name: LINKERD2_PROXY_{{snakecase $proxyK | upper}}_{{snakecase $scopeK | upper}}_{{snakecase $protoK | upper}}_{{snakecase $paramK | upper}} + value: {{ quote $paramV }} +{{ end -}} +{{ end -}} +{{ end -}} +{{ end -}} +{{ if .Values.proxy.opaquePorts -}} +- name: LINKERD2_PROXY_INBOUND_PORTS_DISABLE_PROTOCOL_DETECTION + value: {{.Values.proxy.opaquePorts | quote}} +{{ end -}} +- name: LINKERD2_PROXY_DESTINATION_CONTEXT + value: | + {"ns":"$(_pod_ns)", "nodeName":"$(_pod_nodeName)", "pod":"$(_pod_name)"} +- name: _pod_sa + valueFrom: + fieldRef: + fieldPath: spec.serviceAccountName +- name: _l5d_ns + value: {{.Release.Namespace}} +- name: _l5d_trustdomain + value: {{$trustDomain}} +- name: LINKERD2_PROXY_IDENTITY_DIR + value: /var/run/linkerd/identity/end-entity +- name: LINKERD2_PROXY_IDENTITY_TRUST_ANCHORS +{{- /* +Pods in the `linkerd` namespace are not injected by the proxy injector and instead obtain +the trust anchor bundle from the `linkerd-identity-trust-roots` configmap. This should not +be used in other contexts. +*/}} +{{- if .Values.proxy.loadTrustBundleFromConfigMap }} + valueFrom: + configMapKeyRef: + name: linkerd-identity-trust-roots + key: ca-bundle.crt +{{ else }} + value: | + {{- required "Please provide the identity trust anchors" .Values.identityTrustAnchorsPEM | trim | nindent 4 }} +{{ end -}} +- name: LINKERD2_PROXY_IDENTITY_TOKEN_FILE +{{- if .Values.identity.serviceAccountTokenProjection }} + value: /var/run/secrets/tokens/linkerd-identity-token +{{ else }} + value: /var/run/secrets/kubernetes.io/serviceaccount/token +{{ end -}} +- name: LINKERD2_PROXY_IDENTITY_SVC_ADDR + value: {{ternary "localhost.:8080" (printf "linkerd-identity-headless.%s.svc.%s.:8080" .Release.Namespace .Values.clusterDomain) (eq (toString .Values.proxy.component) "linkerd-identity")}} +- name: LINKERD2_PROXY_IDENTITY_LOCAL_NAME + value: $(_pod_sa).$(_pod_ns).serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_IDENTITY_SVC_NAME + value: linkerd-identity.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_DESTINATION_SVC_NAME + value: linkerd-destination.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +- name: LINKERD2_PROXY_POLICY_SVC_NAME + value: linkerd-destination.{{.Release.Namespace}}.serviceaccount.identity.{{.Release.Namespace}}.{{$trustDomain}} +{{ if .Values.proxy.accessLog -}} +- name: LINKERD2_PROXY_ACCESS_LOG + value: {{.Values.proxy.accessLog | quote}} +{{ end -}} +{{ if .Values.proxy.shutdownGracePeriod -}} +- name: LINKERD2_PROXY_SHUTDOWN_GRACE_PERIOD + value: {{.Values.proxy.shutdownGracePeriod | quote}} +{{ end -}} +{{ if .Values.proxy.additionalEnv -}} +{{ toYaml .Values.proxy.additionalEnv }} +{{ end -}} +{{ if .Values.proxy.experimentalEnv -}} +{{ toYaml .Values.proxy.experimentalEnv }} +{{ end -}} +image: {{.Values.proxy.image.name}}:{{.Values.proxy.image.version | default .Values.linkerdVersion}} +imagePullPolicy: {{.Values.proxy.image.pullPolicy | default .Values.imagePullPolicy}} +livenessProbe: + httpGet: + path: /live + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.livenessProbe.initialDelaySeconds }} + timeoutSeconds: {{.Values.proxy.livenessProbe.timeoutSeconds }} +name: linkerd-proxy +ports: +- containerPort: {{.Values.proxy.ports.inbound}} + name: linkerd-proxy +- containerPort: {{.Values.proxy.ports.admin}} + name: linkerd-admin +readinessProbe: + httpGet: + path: /ready + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.readinessProbe.initialDelaySeconds }} + timeoutSeconds: {{.Values.proxy.readinessProbe.timeoutSeconds }} +{{- if and .Values.proxy.nativeSidecar .Values.proxy.await }} +startupProbe: + httpGet: + path: /ready + port: {{.Values.proxy.ports.admin}} + initialDelaySeconds: {{.Values.proxy.startupProbe.initialDelaySeconds}} + periodSeconds: {{.Values.proxy.startupProbe.periodSeconds}} + failureThreshold: {{.Values.proxy.startupProbe.failureThreshold}} +{{- end }} +{{- if .Values.proxy.resources }} +{{ include "partials.resources" .Values.proxy.resources }} +{{- end }} +securityContext: + allowPrivilegeEscalation: false + {{- if .Values.proxy.capabilities -}} + {{- include "partials.proxy.capabilities" . | nindent 2 -}} + {{- end }} + readOnlyRootFilesystem: true + runAsNonRoot: true + runAsUser: {{.Values.proxy.uid}} +{{- if ge (int .Values.proxy.gid) 0 }} + runAsGroup: {{.Values.proxy.gid}} +{{- end }} + seccompProfile: + type: RuntimeDefault +terminationMessagePolicy: FallbackToLogsOnError +{{- if and (not .Values.proxy.nativeSidecar) (or .Values.proxy.await .Values.proxy.waitBeforeExitSeconds) }} +lifecycle: +{{- if .Values.proxy.await }} + postStart: + exec: + command: + - /usr/lib/linkerd/linkerd-await + - --timeout=2m + - --port={{.Values.proxy.ports.admin}} +{{- end }} +{{- if .Values.proxy.waitBeforeExitSeconds }} + preStop: + exec: + command: + - /bin/sleep + - {{.Values.proxy.waitBeforeExitSeconds | quote}} +{{- end }} +{{- end }} +volumeMounts: +- mountPath: /var/run/linkerd/identity/end-entity + name: linkerd-identity-end-entity +{{- if .Values.identity.serviceAccountTokenProjection }} +- mountPath: /var/run/secrets/tokens + name: linkerd-identity-token +{{- end }} +{{- if .Values.proxy.saMountPath }} +- mountPath: {{.Values.proxy.saMountPath.mountPath}} + name: {{.Values.proxy.saMountPath.name}} + readOnly: {{.Values.proxy.saMountPath.readOnly}} +{{- end -}} +{{- if .Values.proxy.nativeSidecar }} +restartPolicy: Always +{{- end -}} +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_pull-secrets.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_pull-secrets.tpl new file mode 100644 index 0000000000..0c9aa4f01c --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_pull-secrets.tpl @@ -0,0 +1,6 @@ +{{- define "partials.image-pull-secrets"}} +{{- if . }} +imagePullSecrets: +{{ toYaml . | indent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_resources.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_resources.tpl new file mode 100644 index 0000000000..1fd6789fd7 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_resources.tpl @@ -0,0 +1,28 @@ +{{- define "partials.resources" -}} +{{- $ephemeralStorage := index . "ephemeral-storage" -}} +resources: + {{- if or (.cpu).limit (.memory).limit ($ephemeralStorage).limit }} + limits: + {{- with (.cpu).limit }} + cpu: {{. | quote}} + {{- end }} + {{- with (.memory).limit }} + memory: {{. | quote}} + {{- end }} + {{- with ($ephemeralStorage).limit }} + ephemeral-storage: {{. | quote}} + {{- end }} + {{- end }} + {{- if or (.cpu).request (.memory).request ($ephemeralStorage).request }} + requests: + {{- with (.cpu).request }} + cpu: {{. | quote}} + {{- end }} + {{- with (.memory).request }} + memory: {{. | quote}} + {{- end }} + {{- with ($ephemeralStorage).request }} + ephemeral-storage: {{. | quote}} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_tolerations.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_tolerations.tpl new file mode 100644 index 0000000000..c2292b1464 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_tolerations.tpl @@ -0,0 +1,4 @@ +{{- define "linkerd.tolerations" -}} +tolerations: +{{ toYaml .Values.tolerations | trim | indent 2 }} +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_trace.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_trace.tpl new file mode 100644 index 0000000000..dee059541f --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_trace.tpl @@ -0,0 +1,5 @@ +{{ define "partials.linkerd.trace" -}} +{{ if .Values.controlPlaneTracing -}} +- -trace-collector=collector.{{.Values.controlPlaneTracingNamespace}}.svc.{{.Values.clusterDomain}}:55678 +{{ end -}} +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_validate.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_validate.tpl new file mode 100644 index 0000000000..ba772c2fee --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_validate.tpl @@ -0,0 +1,19 @@ +{{- define "linkerd.webhook.validation" -}} + +{{- if and (.injectCaFrom) (.injectCaFromSecret) -}} +{{- fail "injectCaFrom and injectCaFromSecret cannot both be set" -}} +{{- end -}} + +{{- if and (or (.injectCaFrom) (.injectCaFromSecret)) (.caBundle) -}} +{{- fail "injectCaFrom or injectCaFromSecret cannot be set if providing a caBundle" -}} +{{- end -}} + +{{- if and (.externalSecret) (empty .caBundle) (empty .injectCaFrom) (empty .injectCaFromSecret) -}} +{{- fail "if externalSecret is set, then caBundle, injectCaFrom, or injectCaFromSecret must be set" -}} +{{- end }} + +{{- if and (or .injectCaFrom .injectCaFromSecret .caBundle) (not .externalSecret) -}} +{{- fail "if caBundle, injectCaFrom, or injectCaFromSecret is set, then externalSecret must be set" -}} +{{- end -}} + +{{- end -}} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_volumes.tpl b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_volumes.tpl new file mode 100644 index 0000000000..ecb24cfe63 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/templates/_volumes.tpl @@ -0,0 +1,41 @@ +{{ define "partials.proxy.volumes.identity" -}} +emptyDir: + medium: Memory +name: linkerd-identity-end-entity +{{- end -}} + +{{ define "partials.proxyInit.volumes.xtables" -}} +emptyDir: {} +name: {{ .Values.proxyInit.xtMountPath.name }} +{{- end -}} + +{{- define "partials.proxy.volumes.service-account-token" -}} +name: linkerd-identity-token +projected: + sources: + - serviceAccountToken: + path: linkerd-identity-token + expirationSeconds: 86400 {{- /* # 24 hours */}} + audience: identity.l5d.io +{{- end -}} + +{{- define "partials.volumes.manual-mount-service-account-token" -}} +name: kube-api-access +projected: + defaultMode: 420 + sources: + - serviceAccountToken: + expirationSeconds: 3607 + path: token + - configMap: + items: + - key: ca.crt + path: ca.crt + name: kube-root-ca.crt + - downwardAPI: + items: + - fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + path: namespace +{{- end -}} \ No newline at end of file diff --git a/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/values.yaml b/charts/buoyant/linkerd-crds/2024.11.3/charts/partials/values.yaml new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/NOTES.txt b/charts/buoyant/linkerd-crds/2024.11.3/templates/NOTES.txt new file mode 100644 index 0000000000..4ff5c1818a --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/NOTES.txt @@ -0,0 +1,6 @@ +The linkerd-crds chart was successfully installed 🎉 + +To complete the linkerd core installation, please now proceed to install the +linkerd-control-plane chart in the {{ .Release.Namespace }} namespace. + +Looking for more? Visit https://linkerd.io/2/getting-started/ diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_grpcroutes.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_grpcroutes.yaml new file mode 100644 index 0000000000..0050aac88b --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_grpcroutes.yaml @@ -0,0 +1,1507 @@ +{{- if .Values.enableHttpRoutes }} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/1923 + gateway.networking.k8s.io/bundle-version: v0.7.1 + gateway.networking.k8s.io/channel: experimental + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} + creationTimestamp: null + name: grpcroutes.gateway.networking.k8s.io +spec: + group: gateway.networking.k8s.io + names: + categories: + - gateway-api + kind: GRPCRoute + listKind: GRPCRouteList + plural: grpcroutes + singular: grpcroute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha2 + schema: + openAPIV3Schema: + description: "GRPCRoute provides a way to route gRPC requests. This includes + the capability to match requests by hostname, gRPC service, gRPC method, + or HTTP/2 header. Filters can be used to specify additional processing steps. + Backends specify where matching requests will be routed. \n GRPCRoute falls + under extended support within the Gateway API. Within the following specification, + the word \"MUST\" indicates that an implementation supporting GRPCRoute + must conform to the indicated requirement, but an implementation not supporting + this route type need not follow the requirement unless explicitly indicated. + \n Implementations supporting `GRPCRoute` with the `HTTPS` `ProtocolType` + MUST accept HTTP/2 connections without an initial upgrade from HTTP/1.1, + i.e. via ALPN. If the implementation does not support this, then it MUST + set the \"Accepted\" condition to \"False\" for the affected listener with + a reason of \"UnsupportedProtocol\". Implementations MAY also accept HTTP/2 + connections with an upgrade from HTTP/1. \n Implementations supporting `GRPCRoute` + with the `HTTP` `ProtocolType` MUST support HTTP/2 over cleartext TCP (h2c, + https://www.rfc-editor.org/rfc/rfc7540#section-3.1) without an initial upgrade + from HTTP/1.1, i.e. with prior knowledge (https://www.rfc-editor.org/rfc/rfc7540#section-3.4). + If the implementation does not support this, then it MUST set the \"Accepted\" + condition to \"False\" for the affected listener with a reason of \"UnsupportedProtocol\". + Implementations MAY also accept HTTP/2 connections with an upgrade from + HTTP/1, i.e. without prior knowledge. \n Support: Extended" + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of GRPCRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostnames to match against + the GRPC Host header to select a GRPCRoute to process the request. + This matches the RFC 1123 definition of a hostname with 2 notable + exceptions: \n 1. IPs are not allowed. 2. A hostname may be prefixed + with a wildcard label (`*.`). The wildcard label MUST appear by + itself as the first label. \n If a hostname is specified by both + the Listener and GRPCRoute, there MUST be at least one intersecting + hostname for the GRPCRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + GRPCRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches GRPCRoutes + that have either not specified any hostnames or have specified at + least one hostname that matches the Listener hostname. For example, + `test.example.com` and `*.example.com` would both match. On the + other hand, `example.com` and `test.example.net` would not match. + \n Hostnames that are prefixed with a wildcard label (`*.`) are + interpreted as a suffix match. That means that a match for `*.example.com` + would match both `test.example.com`, and `foo.test.example.com`, + but not `example.com`. \n If both the Listener and GRPCRoute have + specified hostnames, any GRPCRoute hostnames that do not match the + Listener hostname MUST be ignored. For example, if a Listener specified + `*.example.com`, and the GRPCRoute specified `test.example.com` + and `test.example.net`, `test.example.net` MUST NOT be considered + for a match. \n If both the Listener and GRPCRoute have specified + hostnames, and none match with the criteria above, then the GRPCRoute + MUST NOT be accepted by the implementation. The implementation MUST + raise an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n If a Route (A) of type HTTPRoute or GRPCRoute + is attached to a Listener and that listener already has another + Route (B) of the other type attached and the intersection of the + hostnames of A and B is non-empty, then the implementation MUST + accept exactly one of these two routes, determined by the following + criteria, in order: \n * The oldest Route based on creation timestamp. + * The Route appearing first in alphabetical order by \"{namespace}/{name}\". + \n The rejected Route MUST raise an 'Accepted' condition with a + status of 'False' in the corresponding RouteParentStatus. \n Support: + Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged. \n Note that for ParentRefs that cross namespace + boundaries, there are specific rules. Cross-namespace references + are only valid if they are explicitly allowed by something in the + namespace they are referring to. For example, Gateway has the AllowedRoutes + field, and ReferenceGrant provides a generic way to enable any other + kind of cross-namespace reference." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the core + API group (such as for a \"Service\" kind referent), Group + must be explicitly set to \"\" (empty string). \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) \n Support: Implementation-specific (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified, this refers to the local namespace of the Route. + \n Note that there are specific rules for ParentRefs which + cross namespace boundaries. Cross-namespace references are + only valid if they are explicitly allowed by something in + the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides a + generic way to enable any other kind of cross-namespace reference. + \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. It + can be interpreted differently based on the type of parent + resource. \n When the parent resource is a Gateway, this targets + all listeners listening on the specified port that also support + this kind of Route(and select this Route). It's not recommended + to set `Port` unless the networking behaviors specified in + a Route must apply to a specific port as opposed to a listener(s) + whose port(s) may be changed. When both Port and SectionName + are specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY choose + to support other parent resources. Implementations supporting + other types of parent resources MUST clearly document how/if + Port is interpreted. \n For the purpose of status, an attachment + is considered successful as long as the parent resource accepts + it partially. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - method: + type: Exact + description: Rules are a list of GRPC matchers, filters and actions. + items: + description: GRPCRouteRule defines the semantics for matching a + gRPC request based on conditions (matches), processing it (filters), + and forwarding the request to an API object (backendRefs). + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. \n Failure behavior here depends + on how many BackendRefs are specified and how many are invalid. + \n If *all* entries in BackendRefs are invalid, and there + are also no filters specified in this route rule, *all* traffic + which matches this rule MUST receive an `UNAVAILABLE` status. + \n See the GRPCBackendRef definition for the rules about what + makes a single GRPCBackendRef invalid. \n When a GRPCBackendRef + is invalid, `UNAVAILABLE` statuses MUST be returned for requests + that would have otherwise been routed to an invalid backend. + If multiple backends are specified, and some are invalid, + the proportion of requests that would otherwise have been + routed to an invalid backend MUST receive an `UNAVAILABLE` + status. \n For example, if two backends are specified with + equal weights, and one is invalid, 50 percent of traffic MUST + receive an `UNAVAILABLE` status. Implementations may choose + how that 50 percent is determined. \n Support: Core for Kubernetes + Service \n Support: Implementation-specific for any other + resource \n Support for weight: Core" + items: + description: GRPCBackendRef defines how a GRPCRoute forwards + a gRPC request. + properties: + filters: + description: "Filters defined at this level MUST be executed + if and only if the request is being forwarded to the + backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in GRPCRouteRule.)" + items: + description: GRPCRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. GRPCRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core API + group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For + example "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for + a filter that mirrors requests. Requests are sent + to the specified destination, but responses from + that destination are ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource + where mirrored requests are sent. \n If the + referent cannot be found, this BackendRef + is invalid and must be dropped from the Gateway. + The controller must ensure the \"ResolvedRefs\" + condition on the Route status is set to `status: + False` and not configure this backend in the + underlying implementation. \n If there is + a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: + False`, with the \"RefNotPermitted\" reason + and not configure this backend in the underlying + implementation. \n In either error case, the + Message of the `ResolvedRefs` Condition should + be used to provide more detail about the problem. + \n Support: Extended for Kubernetes Service + \n Support: Implementation-specific for any + other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core + API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to + CNAME DNS records that may live outside + of the cluster and as such are difficult + to reason about in terms of conformance. + They also may not be safe to forward to + (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with + a type other than ExternalName) \n Support: + Implementation-specific (Services with + type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace + of the backend. When unspecified, the + local namespace is inferred. \n Note that + when a namespace different than the local + namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept + the reference. See the ReferenceGrant + documentation for details. \n Support: + Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination + port number to use for this resource. + Port is required when the referent is + a Kubernetes Service. In this case, the + port number is the service port number, + not the target port. For other resources, + destination port might be derived from + the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + supporting GRPCRoute MUST support core filters. + \n - Extended: Filter types and their corresponding + configuration defined by \"Support: Extended\" + in this package, e.g. \"RequestMirror\". Implementers + are encouraged to support extended filters. \n + - Implementation-specific: Filters that are defined + and supported by specific vendors. In the future, + filters showing convergence in behavior across + multiple implementations will be considered for + inclusion in extended or core conformance levels. + Filter-specific configuration for such filters + is specified using the ExtensionRef field. `Type` + MUST be set to \"ExtensionRef\" for custom filters. + \n Implementers are encouraged to define custom + implementation types to extend the core API with + implementation-specific behavior. \n If a reference + to a custom filter type cannot be resolved, the + filter MUST NOT be skipped. Instead, requests + that would have been processed by that filter + MUST receive a HTTP error response. \n " + enum: + - ResponseHeaderModifier + - RequestHeaderModifier + - RequestMirror + - ExtensionRef + type: string + required: + - type + type: object + maxItems: 16 + type: array + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource kind of + the referent. For example \"Service\". \n Defaults to + \"Service\" when not specified. \n ExternalName services + can refer to CNAME DNS records that may live outside + of the cluster and as such are difficult to reason about + in terms of conformance. They also may not be safe to + forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName Services. + \n Support: Core (Services with a type other than ExternalName) + \n Support: Implementation-specific (Services with type + ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace different than the local + namespace is specified, a ReferenceGrant object is required + in the referent namespace to allow that namespace's + owner to accept the reference. See the ReferenceGrant + documentation for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + required: + - name + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations that support GRPCRoute. - Implementers + are encouraged to support extended filters. - Implementation-specific + custom filters have no API guarantees across implementations. + \n Specifying a core filter multiple times has unspecified + or implementation-specific conformance. Support: Core" + items: + description: GRPCRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + GRPCRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. For + example, "gateway.networking.k8s.io". When unspecified + or empty string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For example + "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for a filter + that mirrors requests. Requests are sent to the specified + destination, but responses from that destination are + ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource where + mirrored requests are sent. \n If the referent cannot + be found, this BackendRef is invalid and must be + dropped from the Gateway. The controller must ensure + the \"ResolvedRefs\" condition on the Route status + is set to `status: False` and not configure this + backend in the underlying implementation. \n If + there is a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: False`, + with the \"RefNotPermitted\" reason and not configure + this backend in the underlying implementation. \n + In either error case, the Message of the `ResolvedRefs` + Condition should be used to provide more detail + about the problem. \n Support: Extended for Kubernetes + Service \n Support: Implementation-specific for + any other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". When + unspecified or empty string, core API group + is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to CNAME + DNS records that may live outside of the cluster + and as such are difficult to reason about in + terms of conformance. They also may not be safe + to forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with a + type other than ExternalName) \n Support: Implementation-specific + (Services with type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the + backend. When unspecified, the local namespace + is inferred. \n Note that when a namespace different + than the local namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept the + reference. See the ReferenceGrant documentation + for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port + number to use for this resource. Port is required + when the referent is a Kubernetes Service. In + this case, the port number is the service port + number, not the target port. For other resources, + destination port might be derived from the referent + resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n Support: + Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\". + All implementations supporting GRPCRoute MUST support + core filters. \n - Extended: Filter types and their + corresponding configuration defined by \"Support: Extended\" + in this package, e.g. \"RequestMirror\". Implementers + are encouraged to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific vendors. + In the future, filters showing convergence in behavior + across multiple implementations will be considered for + inclusion in extended or core conformance levels. Filter-specific + configuration for such filters is specified using the + ExtensionRef field. `Type` MUST be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged to + define custom implementation types to extend the core + API with implementation-specific behavior. \n If a reference + to a custom filter type cannot be resolved, the filter + MUST NOT be skipped. Instead, requests that would have + been processed by that filter MUST receive a HTTP error + response. \n " + enum: + - ResponseHeaderModifier + - RequestHeaderModifier + - RequestMirror + - ExtensionRef + type: string + required: + - type + type: object + maxItems: 16 + type: array + matches: + description: "Matches define conditions used for matching the + rule against incoming gRPC requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - method: service: foo.bar headers: values: + version: 2 - method: service: foo.bar.v2 ``` \n For a request + to match against this rule, it MUST satisfy EITHER of the + two conditions: \n - service of foo.bar AND contains the header + `version: 2` - service of foo.bar.v2 \n See the documentation + for GRPCRouteMatch on how to specify multiple match conditions + to be ANDed together. \n If no matches are specified, the + implementation MUST match every gRPC request. \n Proxy or + Load Balancer routing configuration generated from GRPCRoutes + MUST prioritize rules based on the following criteria, continuing + on ties. Merging MUST not be done between GRPCRoutes and HTTPRoutes. + Precedence MUST be given to the rule with the largest number + of: \n * Characters in a matching non-wildcard hostname. * + Characters in a matching hostname. * Characters in a matching + service. * Characters in a matching method. * Header matches. + \n If ties still exist across multiple Routes, matching precedence + MUST be determined in order of the following criteria, continuing + on ties: \n * The oldest Route based on creation timestamp. + * The Route appearing first in alphabetical order by \"{namespace}/{name}\". + \n If ties still exist within the Route that has been given + precedence, matching precedence MUST be granted to the first + matching rule meeting the above criteria." + items: + description: "GRPCRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a gRPC request only if its service is `foo` + AND it contains the `version: v1` header: \n ``` matches: + - method: type: Exact service: \"foo\" headers: - name: + \"version\" value \"v1\" \n ```" + properties: + headers: + description: Headers specifies gRPC request header matchers. + Multiple match values are ANDed together, meaning, a + request MUST match all the specified headers to select + the route. + items: + description: GRPCHeaderMatch describes how to select + a gRPC route by matching gRPC request headers. + properties: + name: + description: "Name is the name of the gRPC Header + to be matched. \n If multiple entries specify + equivalent header names, only the first entry + with an equivalent name MUST be considered for + a match. Subsequent entries with an equivalent + header name MUST be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: Type specifies how to match against + the value of the header. + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of the gRPC Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: Method specifies a gRPC request service/method + matcher. If this field is not specified, all services + and methods will match. + properties: + method: + description: "Value of the method to match against. + If left empty or omitted, will match all services. + \n At least one of Service and Method MUST be a + non-empty string." + maxLength: 1024 + type: string + service: + description: "Value of the service to match against. + If left empty or omitted, will match any service. + \n At least one of Service and Method MUST be a + non-empty string." + maxLength: 1024 + type: string + type: + default: Exact + description: "Type specifies how to match against + the service and/or method. Support: Core (Exact + with service and method specified) \n Support: Implementation-specific + (Exact with method specified but no service specified) + \n Support: Implementation-specific (RegularExpression)" + enum: + - Exact + - RegularExpression + type: string + type: object + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of GRPCRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, \n type FooStatus struct{ + // Represents the observations of a foo's current state. + // Known .status.conditions.type are: \"Available\", \"Progressing\", + and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge + // +listType=map // +listMapKey=type Conditions []metav1.Condition + `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" + protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields + }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the + core API group (such as for a \"Service\" kind referent), + Group must be explicitly set to \"\" (empty string). \n + Support: Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) \n Support: Implementation-specific (Other + Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified, this refers to the local namespace of + the Route. \n Note that there are specific rules for ParentRefs + which cross namespace boundaries. Cross-namespace references + are only valid if they are explicitly allowed by something + in the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides + a generic way to enable any other kind of cross-namespace + reference. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: null + storedVersions: null +{{- end }} + diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_httproutes.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_httproutes.yaml new file mode 100644 index 0000000000..b695c51d50 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_httproutes.yaml @@ -0,0 +1,3881 @@ +{{- if .Values.enableHttpRoutes }} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/1923 + gateway.networking.k8s.io/bundle-version: v0.7.1 + gateway.networking.k8s.io/channel: experimental + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} + creationTimestamp: null + name: httproutes.gateway.networking.k8s.io +spec: + group: gateway.networking.k8s.io + names: + categories: + - gateway-api + kind: HTTPRoute + listKind: HTTPRouteList + plural: httproutes + singular: httproute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: The v1alpha2 version of HTTPRoute has been deprecated and + will be removed in a future release of the API. Please upgrade to v1beta1. + name: v1alpha2 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute used to process + the request. Implementations MUST ignore any port value specified + in the HTTP Host header while performing a match. \n Valid values + for Hostnames are determined by RFC 1123 definition of a hostname + with 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + that have either not specified any hostnames or have specified at + least one hostname that matches the Listener hostname. For example, + `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n In the event that multiple HTTPRoutes specify + intersecting hostnames (e.g. overlapping wildcard matching and exact + matching hostnames), precedence must be given to rules from the + HTTPRoute with the largest number of: \n * Characters in a matching + non-wildcard hostname. * Characters in a matching hostname. \n If + ties exist across multiple Routes, the matching precedence rules + for HTTPRouteMatches takes over. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged. \n Note that for ParentRefs that cross namespace + boundaries, there are specific rules. Cross-namespace references + are only valid if they are explicitly allowed by something in the + namespace they are referring to. For example, Gateway has the AllowedRoutes + field, and ReferenceGrant provides a generic way to enable any other + kind of cross-namespace reference." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the core + API group (such as for a \"Service\" kind referent), Group + must be explicitly set to \"\" (empty string). \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) \n Support: Implementation-specific (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified, this refers to the local namespace of the Route. + \n Note that there are specific rules for ParentRefs which + cross namespace boundaries. Cross-namespace references are + only valid if they are explicitly allowed by something in + the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides a + generic way to enable any other kind of cross-namespace reference. + \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. It + can be interpreted differently based on the type of parent + resource. \n When the parent resource is a Gateway, this targets + all listeners listening on the specified port that also support + this kind of Route(and select this Route). It's not recommended + to set `Port` unless the networking behaviors specified in + a Route must apply to a specific port as opposed to a listener(s) + whose port(s) may be changed. When both Port and SectionName + are specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY choose + to support other parent resources. Implementations supporting + other types of parent resources MUST clearly document how/if + Port is interpreted. \n For the purpose of status, an attachment + is considered successful as long as the parent resource accepts + it partially. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches), processing it (filters), + and forwarding the request to an API object (backendRefs). + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. \n Failure behavior here depends + on how many BackendRefs are specified and how many are invalid. + \n If *all* entries in BackendRefs are invalid, and there + are also no filters specified in this route rule, *all* traffic + which matches this rule MUST receive a 500 status code. \n + See the HTTPBackendRef definition for the rules about what + makes a single HTTPBackendRef invalid. \n When a HTTPBackendRef + is invalid, 500 status codes MUST be returned for requests + that would have otherwise been routed to an invalid backend. + If multiple backends are specified, and some are invalid, + the proportion of requests that would otherwise have been + routed to an invalid backend MUST receive a 500 status code. + \n For example, if two backends are specified with equal weights, + and one is invalid, 50 percent of traffic must receive a 500. + Implementations may choose how that 50 percent is determined. + \n Support: Core for Kubernetes Service \n Support: Extended + for Kubernetes ServiceImport \n Support: Implementation-specific + for any other resource \n Support for weight: Core" + items: + description: HTTPBackendRef defines how a HTTPRoute should + forward an HTTP request. + properties: + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core API + group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For + example "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for + a filter that mirrors requests. Requests are sent + to the specified destination, but responses from + that destination are ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource + where mirrored requests are sent. \n If the + referent cannot be found, this BackendRef + is invalid and must be dropped from the Gateway. + The controller must ensure the \"ResolvedRefs\" + condition on the Route status is set to `status: + False` and not configure this backend in the + underlying implementation. \n If there is + a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: + False`, with the \"RefNotPermitted\" reason + and not configure this backend in the underlying + implementation. \n In either error case, the + Message of the `ResolvedRefs` Condition should + be used to provide more detail about the problem. + \n Support: Extended for Kubernetes Service + \n Support: Implementation-specific for any + other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core + API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to + CNAME DNS records that may live outside + of the cluster and as such are difficult + to reason about in terms of conformance. + They also may not be safe to forward to + (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with + a type other than ExternalName) \n Support: + Implementation-specific (Services with + type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace + of the backend. When unspecified, the + local namespace is inferred. \n Note that + when a namespace different than the local + namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept + the reference. See the ReferenceGrant + documentation for details. \n Support: + Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination + port number to use for this resource. + Port is required when the referent is + a Kubernetes Service. In this case, the + port number is the service port number, + not the target port. For other resources, + destination port might be derived from + the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" would + be modified to \"/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestMirror + - RequestRedirect + - URLRewrite + - ExtensionRef + type: string + urlRewrite: + description: "URLRewrite defines a schema for a + filter that modifies a request during forwarding. + \n Support: Extended" + properties: + hostname: + description: "Hostname is the value to be used + to replace the Host header value during forwarding. + \n Support: Extended" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines a path rewrite. \n + Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" would + be modified to \"/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + type: object + required: + - type + type: object + maxItems: 16 + type: array + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource kind of + the referent. For example \"Service\". \n Defaults to + \"Service\" when not specified. \n ExternalName services + can refer to CNAME DNS records that may live outside + of the cluster and as such are difficult to reason about + in terms of conformance. They also may not be safe to + forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName Services. + \n Support: Core (Services with a type other than ExternalName) + \n Support: Implementation-specific (Services with type + ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace different than the local + namespace is specified, a ReferenceGrant object is required + in the referent namespace to allow that namespace's + owner to accept the reference. See the ReferenceGrant + documentation for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + required: + - name + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or implementation-specific + conformance. \n All filters are expected to be compatible + with each other except for the URLRewrite and RequestRedirect + filters, which may not be combined. If an implementation can + not support other combinations of filters, they must clearly + document that limitation. In all cases where incompatible + or unsupported filters are specified, implementations MUST + add a warning condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. For + example, "gateway.networking.k8s.io". When unspecified + or empty string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For example + "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for a filter + that mirrors requests. Requests are sent to the specified + destination, but responses from that destination are + ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource where + mirrored requests are sent. \n If the referent cannot + be found, this BackendRef is invalid and must be + dropped from the Gateway. The controller must ensure + the \"ResolvedRefs\" condition on the Route status + is set to `status: False` and not configure this + backend in the underlying implementation. \n If + there is a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: False`, + with the \"RefNotPermitted\" reason and not configure + this backend in the underlying implementation. \n + In either error case, the Message of the `ResolvedRefs` + Condition should be used to provide more detail + about the problem. \n Support: Extended for Kubernetes + Service \n Support: Implementation-specific for + any other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". When + unspecified or empty string, core API group + is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to CNAME + DNS records that may live outside of the cluster + and as such are difficult to reason about in + terms of conformance. They also may not be safe + to forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with a + type other than ExternalName) \n Support: Implementation-specific + (Services with type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the + backend. When unspecified, the local namespace + is inferred. \n Note that when a namespace different + than the local namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept the + reference. See the ReferenceGrant documentation + for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port + number to use for this resource. Port is required + when the referent is a Kubernetes Service. In + this case, the port number is the service port + number, not the target port. For other resources, + destination port might be derived from the referent + resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname in the `Host` header of + the request is used. \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to modify + the path of the incoming request. The modified path + is then used to construct the `Location` header. + When empty, the request path is used as-is. \n Support: + Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the value + with which to replace the full path of a request + during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies the + value with which to replace the prefix match + of a request during a rewrite or redirect. For + example, a request to \"/foo/bar\" with a prefix + match of \"/foo\" would be modified to \"/bar\". + \n Note that this matches the behavior of the + PathPrefix match type. This matches full path + elements. A path element refers to the list + of labels in the path split by the `/` separator. + When specified, a trailing `/` is ignored. For + example, the paths `/abc`, `/abc/`, and `/abc/def` + would all match the prefix `/abc`, but the path + `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path modifier. + Additional types may be added in a future release + of the API. \n Note that values may be added + to this enum, implementations must ensure that + unknown values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the Route + to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. \n If + no port is specified, the redirect port MUST be + derived using the following rules: \n * If redirect + scheme is not-empty, the redirect port MUST be the + well-known port associated with the redirect scheme. + Specifically \"http\" to port 80 and \"https\" to + port 443. If the redirect scheme does not have a + well-known port, the listener port of the Gateway + SHOULD be used. * If redirect scheme is empty, the + redirect port MUST be the Gateway Listener port. + \n Implementations SHOULD NOT add the port number + in the 'Location' header in the following cases: + \n * A Location header that will use HTTP (whether + that is determined via the Listener protocol or + the Scheme field) _and_ use port 80. * A Location + header that will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) _and_ + use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Scheme redirects can affect the port of the redirect, + for more information, refer to the documentation + for the port field of this filter. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause a + crash. \n Unknown values here must result in the + implementation setting the Accepted Condition for + the Route to `status: False`, with a Reason of `UnsupportedValue`. + \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. \n Unknown + values here must result in the implementation setting + the Accepted Condition for the Route to `status: + False`, with a Reason of `UnsupportedValue`. \n + Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n Support: + Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\". + All implementations must support core filters. \n - + Extended: Filter types and their corresponding configuration + defined by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged to support + extended filters. \n - Implementation-specific: Filters + that are defined and supported by specific vendors. + In the future, filters showing convergence in behavior + across multiple implementations will be considered for + inclusion in extended or core conformance levels. Filter-specific + configuration for such filters is specified using the + ExtensionRef field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged to + define custom implementation types to extend the core + API with implementation-specific behavior. \n If a reference + to a custom filter type cannot be resolved, the filter + MUST NOT be skipped. Instead, requests that would have + been processed by that filter MUST receive a HTTP error + response. \n Note that values may be added to this enum, + implementations must ensure that unknown values will + not cause a crash. \n Unknown values here must result + in the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestMirror + - RequestRedirect + - URLRewrite + - ExtensionRef + type: string + urlRewrite: + description: "URLRewrite defines a schema for a filter + that modifies a request during forwarding. \n Support: + Extended" + properties: + hostname: + description: "Hostname is the value to be used to + replace the Host header value during forwarding. + \n Support: Extended" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines a path rewrite. \n Support: + Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the value + with which to replace the full path of a request + during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies the + value with which to replace the prefix match + of a request during a rewrite or redirect. For + example, a request to \"/foo/bar\" with a prefix + match of \"/foo\" would be modified to \"/bar\". + \n Note that this matches the behavior of the + PathPrefix match type. This matches full path + elements. A path element refers to the list + of labels in the path split by the `/` separator. + When specified, a trailing `/` is ignored. For + example, the paths `/abc`, `/abc/`, and `/abc/def` + would all match the prefix `/abc`, but the path + `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path modifier. + Additional types may be added in a future release + of the API. \n Note that values may be added + to this enum, implementations must ensure that + unknown values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the Route + to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + type: object + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - name: \"version\" + value: \"v2\" - path: value: \"/v2/foo\" ``` \n For a request + to match against this rule, a request must satisfy EITHER + of the two conditions: \n - path prefixed with `/foo` AND + contains the header `version: v2` - path prefix of `/v2/foo` + \n See the documentation for HTTPRouteMatch on how to specify + multiple match conditions that should be ANDed together. \n + If no matches are specified, the default is a prefix path + match on \"/\", which has the effect of matching every HTTP + request. \n Proxy or Load Balancer routing configuration generated + from HTTPRoutes MUST prioritize matches based on the following + criteria, continuing on ties. Across all rules specified on + applicable Routes, precedence must be given to the match having: + \n * \"Exact\" path match. * \"Prefix\" path match with largest + number of characters. * Method match. * Largest number of + header matches. * Largest number of query param matches. \n + Note: The precedence of RegularExpression path matches are + implementation-specific. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within an HTTPRoute, matching precedence MUST + be granted to the FIRST matching rule (in list order) with + a match meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: \n path: value: \"/foo\" headers: - name: \"version\" + value \"v1\" \n ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Implementation-specific (RegularExpression) + \n Since RegularExpression HeaderMatchType has + implementation-specific conformance, implementations + can support POSIX, PCRE or any other dialects + of regular expressions. Please read the implementation's + documentation to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Implementation-specific (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: "QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. \n Support: Extended" + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: "Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + \n If multiple entries specify equivalent query + param names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent query param name MUST + be ignored. \n If a query param is repeated in + an HTTP request, the behavior is purposely left + undefined, since different data planes have different + capabilities. However, it is *recommended* that + implementations should match against the first + value of the param if the data plane supports + it, as this behavior is expected in other load + balancing contexts outside of the Gateway API. + \n Users SHOULD NOT route traffic based on repeated + query params to guard themselves against potential + differences in the implementations." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Implementation-specific + (RegularExpression) \n Since RegularExpression + QueryParamMatchType has Implementation-specific + conformance, implementations can support POSIX, + PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, \n type FooStatus struct{ + // Represents the observations of a foo's current state. + // Known .status.conditions.type are: \"Available\", \"Progressing\", + and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge + // +listType=map // +listMapKey=type Conditions []metav1.Condition + `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" + protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields + }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the + core API group (such as for a \"Service\" kind referent), + Group must be explicitly set to \"\" (empty string). \n + Support: Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) \n Support: Implementation-specific (Other + Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified, this refers to the local namespace of + the Route. \n Note that there are specific rules for ParentRefs + which cross namespace boundaries. Cross-namespace references + are only valid if they are explicitly allowed by something + in the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides + a generic way to enable any other kind of cross-namespace + reference. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta1 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute used to process + the request. Implementations MUST ignore any port value specified + in the HTTP Host header while performing a match. \n Valid values + for Hostnames are determined by RFC 1123 definition of a hostname + with 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + that have either not specified any hostnames or have specified at + least one hostname that matches the Listener hostname. For example, + `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n In the event that multiple HTTPRoutes specify + intersecting hostnames (e.g. overlapping wildcard matching and exact + matching hostnames), precedence must be given to rules from the + HTTPRoute with the largest number of: \n * Characters in a matching + non-wildcard hostname. * Characters in a matching hostname. \n If + ties exist across multiple Routes, the matching precedence rules + for HTTPRouteMatches takes over. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged. \n Note that for ParentRefs that cross namespace + boundaries, there are specific rules. Cross-namespace references + are only valid if they are explicitly allowed by something in the + namespace they are referring to. For example, Gateway has the AllowedRoutes + field, and ReferenceGrant provides a generic way to enable any other + kind of cross-namespace reference." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the core + API group (such as for a \"Service\" kind referent), Group + must be explicitly set to \"\" (empty string). \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) \n Support: Implementation-specific (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified, this refers to the local namespace of the Route. + \n Note that there are specific rules for ParentRefs which + cross namespace boundaries. Cross-namespace references are + only valid if they are explicitly allowed by something in + the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides a + generic way to enable any other kind of cross-namespace reference. + \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. It + can be interpreted differently based on the type of parent + resource. \n When the parent resource is a Gateway, this targets + all listeners listening on the specified port that also support + this kind of Route(and select this Route). It's not recommended + to set `Port` unless the networking behaviors specified in + a Route must apply to a specific port as opposed to a listener(s) + whose port(s) may be changed. When both Port and SectionName + are specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY choose + to support other parent resources. Implementations supporting + other types of parent resources MUST clearly document how/if + Port is interpreted. \n For the purpose of status, an attachment + is considered successful as long as the parent resource accepts + it partially. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches), processing it (filters), + and forwarding the request to an API object (backendRefs). + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. \n Failure behavior here depends + on how many BackendRefs are specified and how many are invalid. + \n If *all* entries in BackendRefs are invalid, and there + are also no filters specified in this route rule, *all* traffic + which matches this rule MUST receive a 500 status code. \n + See the HTTPBackendRef definition for the rules about what + makes a single HTTPBackendRef invalid. \n When a HTTPBackendRef + is invalid, 500 status codes MUST be returned for requests + that would have otherwise been routed to an invalid backend. + If multiple backends are specified, and some are invalid, + the proportion of requests that would otherwise have been + routed to an invalid backend MUST receive a 500 status code. + \n For example, if two backends are specified with equal weights, + and one is invalid, 50 percent of traffic must receive a 500. + Implementations may choose how that 50 percent is determined. + \n Support: Core for Kubernetes Service \n Support: Extended + for Kubernetes ServiceImport \n Support: Implementation-specific + for any other resource \n Support for weight: Core" + items: + description: HTTPBackendRef defines how a HTTPRoute should + forward an HTTP request. + properties: + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core API + group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For + example "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for + a filter that mirrors requests. Requests are sent + to the specified destination, but responses from + that destination are ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource + where mirrored requests are sent. \n If the + referent cannot be found, this BackendRef + is invalid and must be dropped from the Gateway. + The controller must ensure the \"ResolvedRefs\" + condition on the Route status is set to `status: + False` and not configure this backend in the + underlying implementation. \n If there is + a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: + False`, with the \"RefNotPermitted\" reason + and not configure this backend in the underlying + implementation. \n In either error case, the + Message of the `ResolvedRefs` Condition should + be used to provide more detail about the problem. + \n Support: Extended for Kubernetes Service + \n Support: Implementation-specific for any + other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". + When unspecified or empty string, core + API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to + CNAME DNS records that may live outside + of the cluster and as such are difficult + to reason about in terms of conformance. + They also may not be safe to forward to + (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with + a type other than ExternalName) \n Support: + Implementation-specific (Services with + type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace + of the backend. When unspecified, the + local namespace is inferred. \n Note that + when a namespace different than the local + namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept + the reference. See the ReferenceGrant + documentation for details. \n Support: + Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination + port number to use for this resource. + Port is required when the referent is + a Kubernetes Service. In this case, the + port number is the service port number, + not the target port. For other resources, + destination port might be derived from + the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" would + be modified to \"/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestMirror + - RequestRedirect + - URLRewrite + - ExtensionRef + type: string + urlRewrite: + description: "URLRewrite defines a schema for a + filter that modifies a request during forwarding. + \n Support: Extended" + properties: + hostname: + description: "Hostname is the value to be used + to replace the Host header value during forwarding. + \n Support: Extended" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines a path rewrite. \n + Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" would + be modified to \"/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + type: object + required: + - type + type: object + maxItems: 16 + type: array + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource kind of + the referent. For example \"Service\". \n Defaults to + \"Service\" when not specified. \n ExternalName services + can refer to CNAME DNS records that may live outside + of the cluster and as such are difficult to reason about + in terms of conformance. They also may not be safe to + forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName Services. + \n Support: Core (Services with a type other than ExternalName) + \n Support: Implementation-specific (Services with type + ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace different than the local + namespace is specified, a ReferenceGrant object is required + in the referent namespace to allow that namespace's + owner to accept the reference. See the ReferenceGrant + documentation for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + required: + - name + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or implementation-specific + conformance. \n All filters are expected to be compatible + with each other except for the URLRewrite and RequestRedirect + filters, which may not be combined. If an implementation can + not support other combinations of filters, they must clearly + document that limitation. In all cases where incompatible + or unsupported filters are specified, implementations MUST + add a warning condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + extensionRef: + description: "ExtensionRef is an optional, implementation-specific + extension to the \"filter\" behavior. For example, + resource \"myroutefilter\" in group \"networking.example.net\"). + ExtensionRef MUST NOT be used for core and extended + filters. \n Support: Implementation-specific" + properties: + group: + description: Group is the group of the referent. For + example, "gateway.networking.k8s.io". When unspecified + or empty string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is kind of the referent. For example + "HTTPRoute" or "Service". + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestMirror: + description: "RequestMirror defines a schema for a filter + that mirrors requests. Requests are sent to the specified + destination, but responses from that destination are + ignored. \n Support: Extended" + properties: + backendRef: + description: "BackendRef references a resource where + mirrored requests are sent. \n If the referent cannot + be found, this BackendRef is invalid and must be + dropped from the Gateway. The controller must ensure + the \"ResolvedRefs\" condition on the Route status + is set to `status: False` and not configure this + backend in the underlying implementation. \n If + there is a cross-namespace reference to an *existing* + object that is not allowed by a ReferenceGrant, + the controller must ensure the \"ResolvedRefs\" + \ condition on the Route is set to `status: False`, + with the \"RefNotPermitted\" reason and not configure + this backend in the underlying implementation. \n + In either error case, the Message of the `ResolvedRefs` + Condition should be used to provide more detail + about the problem. \n Support: Extended for Kubernetes + Service \n Support: Implementation-specific for + any other resource" + properties: + group: + default: "" + description: Group is the group of the referent. + For example, "gateway.networking.k8s.io". When + unspecified or empty string, core API group + is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource + kind of the referent. For example \"Service\". + \n Defaults to \"Service\" when not specified. + \n ExternalName services can refer to CNAME + DNS records that may live outside of the cluster + and as such are difficult to reason about in + terms of conformance. They also may not be safe + to forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName + Services. \n Support: Core (Services with a + type other than ExternalName) \n Support: Implementation-specific + (Services with type ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the + backend. When unspecified, the local namespace + is inferred. \n Note that when a namespace different + than the local namespace is specified, a ReferenceGrant + object is required in the referent namespace + to allow that namespace's owner to accept the + reference. See the ReferenceGrant documentation + for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port + number to use for this resource. Port is required + when the referent is a Kubernetes Service. In + this case, the port number is the service port + number, not the target port. For other resources, + destination port might be derived from the referent + resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + required: + - name + type: object + required: + - backendRef + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname in the `Host` header of + the request is used. \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to modify + the path of the incoming request. The modified path + is then used to construct the `Location` header. + When empty, the request path is used as-is. \n Support: + Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the value + with which to replace the full path of a request + during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies the + value with which to replace the prefix match + of a request during a rewrite or redirect. For + example, a request to \"/foo/bar\" with a prefix + match of \"/foo\" would be modified to \"/bar\". + \n Note that this matches the behavior of the + PathPrefix match type. This matches full path + elements. A path element refers to the list + of labels in the path split by the `/` separator. + When specified, a trailing `/` is ignored. For + example, the paths `/abc`, `/abc/`, and `/abc/def` + would all match the prefix `/abc`, but the path + `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path modifier. + Additional types may be added in a future release + of the API. \n Note that values may be added + to this enum, implementations must ensure that + unknown values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the Route + to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. \n If + no port is specified, the redirect port MUST be + derived using the following rules: \n * If redirect + scheme is not-empty, the redirect port MUST be the + well-known port associated with the redirect scheme. + Specifically \"http\" to port 80 and \"https\" to + port 443. If the redirect scheme does not have a + well-known port, the listener port of the Gateway + SHOULD be used. * If redirect scheme is empty, the + redirect port MUST be the Gateway Listener port. + \n Implementations SHOULD NOT add the port number + in the 'Location' header in the following cases: + \n * A Location header that will use HTTP (whether + that is determined via the Listener protocol or + the Scheme field) _and_ use port 80. * A Location + header that will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) _and_ + use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Scheme redirects can affect the port of the redirect, + for more information, refer to the documentation + for the port field of this filter. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause a + crash. \n Unknown values here must result in the + implementation setting the Accepted Condition for + the Route to `status: False`, with a Reason of `UnsupportedValue`. + \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. \n Unknown + values here must result in the implementation setting + the Accepted Condition for the Route to `status: + False`, with a Reason of `UnsupportedValue`. \n + Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n Support: + Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: foo + \n Config: add: - name: \"my-header\" value: \"bar,baz\" + \n Output: GET /foo HTTP/1.1 my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo my-header2: + bar my-header3: baz \n Config: remove: [\"my-header1\", + \"my-header3\"] \n Output: GET /foo HTTP/1.1 my-header2: + bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + set: - name: \"my-header\" value: \"bar\" \n Output: + GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\". + All implementations must support core filters. \n - + Extended: Filter types and their corresponding configuration + defined by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged to support + extended filters. \n - Implementation-specific: Filters + that are defined and supported by specific vendors. + In the future, filters showing convergence in behavior + across multiple implementations will be considered for + inclusion in extended or core conformance levels. Filter-specific + configuration for such filters is specified using the + ExtensionRef field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged to + define custom implementation types to extend the core + API with implementation-specific behavior. \n If a reference + to a custom filter type cannot be resolved, the filter + MUST NOT be skipped. Instead, requests that would have + been processed by that filter MUST receive a HTTP error + response. \n Note that values may be added to this enum, + implementations must ensure that unknown values will + not cause a crash. \n Unknown values here must result + in the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestMirror + - RequestRedirect + - URLRewrite + - ExtensionRef + type: string + urlRewrite: + description: "URLRewrite defines a schema for a filter + that modifies a request during forwarding. \n Support: + Extended" + properties: + hostname: + description: "Hostname is the value to be used to + replace the Host header value during forwarding. + \n Support: Extended" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines a path rewrite. \n Support: + Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the value + with which to replace the full path of a request + during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies the + value with which to replace the prefix match + of a request during a rewrite or redirect. For + example, a request to \"/foo/bar\" with a prefix + match of \"/foo\" would be modified to \"/bar\". + \n Note that this matches the behavior of the + PathPrefix match type. This matches full path + elements. A path element refers to the list + of labels in the path split by the `/` separator. + When specified, a trailing `/` is ignored. For + example, the paths `/abc`, `/abc/`, and `/abc/def` + would all match the prefix `/abc`, but the path + `/abcd` would not." + maxLength: 1024 + type: string + type: + description: "Type defines the type of path modifier. + Additional types may be added in a future release + of the API. \n Note that values may be added + to this enum, implementations must ensure that + unknown values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the Route + to `status: False`, with a Reason of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + type: object + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - name: \"version\" + value: \"v2\" - path: value: \"/v2/foo\" ``` \n For a request + to match against this rule, a request must satisfy EITHER + of the two conditions: \n - path prefixed with `/foo` AND + contains the header `version: v2` - path prefix of `/v2/foo` + \n See the documentation for HTTPRouteMatch on how to specify + multiple match conditions that should be ANDed together. \n + If no matches are specified, the default is a prefix path + match on \"/\", which has the effect of matching every HTTP + request. \n Proxy or Load Balancer routing configuration generated + from HTTPRoutes MUST prioritize matches based on the following + criteria, continuing on ties. Across all rules specified on + applicable Routes, precedence must be given to the match having: + \n * \"Exact\" path match. * \"Prefix\" path match with largest + number of characters. * Method match. * Largest number of + header matches. * Largest number of query param matches. \n + Note: The precedence of RegularExpression path matches are + implementation-specific. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within an HTTPRoute, matching precedence MUST + be granted to the FIRST matching rule (in list order) with + a match meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: \n path: value: \"/foo\" headers: - name: \"version\" + value \"v1\" \n ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Implementation-specific (RegularExpression) + \n Since RegularExpression HeaderMatchType has + implementation-specific conformance, implementations + can support POSIX, PCRE or any other dialects + of regular expressions. Please read the implementation's + documentation to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Implementation-specific (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: "QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. \n Support: Extended" + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: "Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + \n If multiple entries specify equivalent query + param names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent query param name MUST + be ignored. \n If a query param is repeated in + an HTTP request, the behavior is purposely left + undefined, since different data planes have different + capabilities. However, it is *recommended* that + implementations should match against the first + value of the param if the data plane supports + it, as this behavior is expected in other load + balancing contexts outside of the Gateway API. + \n Users SHOULD NOT route traffic based on repeated + query params to guard themselves against potential + differences in the implementations." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Implementation-specific + (RegularExpression) \n Since RegularExpression + QueryParamMatchType has Implementation-specific + conformance, implementations can support POSIX, + PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, \n type FooStatus struct{ + // Represents the observations of a foo's current state. + // Known .status.conditions.type are: \"Available\", \"Progressing\", + and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge + // +listType=map // +listMapKey=type Conditions []metav1.Condition + `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" + protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields + }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the + core API group (such as for a \"Service\" kind referent), + Group must be explicitly set to \"\" (empty string). \n + Support: Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) \n Support: Implementation-specific (Other + Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified, this refers to the local namespace of + the Route. \n Note that there are specific rules for ParentRefs + which cross namespace boundaries. Cross-namespace references + are only valid if they are explicitly allowed by something + in the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides + a generic way to enable any other kind of cross-namespace + reference. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: null + storedVersions: null +{{- end }} + diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tcproutes.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tcproutes.yaml new file mode 100644 index 0000000000..cb17293b73 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tcproutes.yaml @@ -0,0 +1,533 @@ +{{- if .Values.enableTcpRoutes }} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/1923 + gateway.networking.k8s.io/bundle-version: v0.7.1 + gateway.networking.k8s.io/channel: experimental + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} + creationTimestamp: null + name: tcproutes.gateway.networking.k8s.io +spec: + group: gateway.networking.k8s.io + names: + categories: + - gateway-api + kind: TCPRoute + listKind: TCPRouteList + plural: tcproutes + singular: tcproute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha2 + schema: + openAPIV3Schema: + description: TCPRoute provides a way to route TCP requests. When combined + with a Gateway listener, it can be used to forward connections on the port + specified by the listener to a set of backends specified by the TCPRoute. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of TCPRoute. + properties: + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged. \n Note that for ParentRefs that cross namespace + boundaries, there are specific rules. Cross-namespace references + are only valid if they are explicitly allowed by something in the + namespace they are referring to. For example, Gateway has the AllowedRoutes + field, and ReferenceGrant provides a generic way to enable any other + kind of cross-namespace reference." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the core + API group (such as for a \"Service\" kind referent), Group + must be explicitly set to \"\" (empty string). \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) \n Support: Implementation-specific (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified, this refers to the local namespace of the Route. + \n Note that there are specific rules for ParentRefs which + cross namespace boundaries. Cross-namespace references are + only valid if they are explicitly allowed by something in + the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides a + generic way to enable any other kind of cross-namespace reference. + \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. It + can be interpreted differently based on the type of parent + resource. \n When the parent resource is a Gateway, this targets + all listeners listening on the specified port that also support + this kind of Route(and select this Route). It's not recommended + to set `Port` unless the networking behaviors specified in + a Route must apply to a specific port as opposed to a listener(s) + whose port(s) may be changed. When both Port and SectionName + are specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY choose + to support other parent resources. Implementations supporting + other types of parent resources MUST clearly document how/if + Port is interpreted. \n For the purpose of status, an attachment + is considered successful as long as the parent resource accepts + it partially. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + description: Rules are a list of TCP matchers and actions. + items: + description: TCPRouteRule is the configuration for a given rule. + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. If unspecified or invalid (refers + to a non-existent resource or a Service with no endpoints), + the underlying implementation MUST actively reject connection + attempts to this backend. Connection rejections must respect + weight; if an invalid backend is requested to have 80% of + connections, then 80% of connections must be rejected instead. + \n Support: Core for Kubernetes Service \n Support: Extended + for Kubernetes ServiceImport \n Support: Implementation-specific + for any other resource \n Support for weight: Extended" + items: + description: "BackendRef defines how a Route should forward + a request to a Kubernetes resource. \n Note that when a + namespace different than the local namespace is specified, + a ReferenceGrant object is required in the referent namespace + to allow that namespace's owner to accept the reference. + See the ReferenceGrant documentation for details." + properties: + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource kind of + the referent. For example \"Service\". \n Defaults to + \"Service\" when not specified. \n ExternalName services + can refer to CNAME DNS records that may live outside + of the cluster and as such are difficult to reason about + in terms of conformance. They also may not be safe to + forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName Services. + \n Support: Core (Services with a type other than ExternalName) + \n Support: Implementation-specific (Services with type + ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace different than the local + namespace is specified, a ReferenceGrant object is required + in the referent namespace to allow that namespace's + owner to accept the reference. See the ReferenceGrant + documentation for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + required: + - name + type: object + maxItems: 16 + minItems: 1 + type: array + type: object + maxItems: 16 + minItems: 1 + type: array + required: + - rules + type: object + status: + description: Status defines the current state of TCPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, \n type FooStatus struct{ + // Represents the observations of a foo's current state. + // Known .status.conditions.type are: \"Available\", \"Progressing\", + and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge + // +listType=map // +listMapKey=type Conditions []metav1.Condition + `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" + protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields + }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the + core API group (such as for a \"Service\" kind referent), + Group must be explicitly set to \"\" (empty string). \n + Support: Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) \n Support: Implementation-specific (Other + Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified, this refers to the local namespace of + the Route. \n Note that there are specific rules for ParentRefs + which cross namespace boundaries. Cross-namespace references + are only valid if they are explicitly allowed by something + in the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides + a generic way to enable any other kind of cross-namespace + reference. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: null + storedVersions: null +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tlsroutes.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tlsroutes.yaml new file mode 100644 index 0000000000..d79a19aaef --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/gateway.networking.k8s.io_tlsroutes.yaml @@ -0,0 +1,582 @@ +{{- if .Values.enableTlsRoutes }} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + api-approved.kubernetes.io: https://github.com/kubernetes-sigs/gateway-api/pull/1923 + gateway.networking.k8s.io/bundle-version: v0.7.1 + gateway.networking.k8s.io/channel: experimental + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} + creationTimestamp: null + name: tlsroutes.gateway.networking.k8s.io +spec: + group: gateway.networking.k8s.io + names: + categories: + - gateway-api + kind: TLSRoute + listKind: TLSRouteList + plural: tlsroutes + singular: tlsroute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha2 + schema: + openAPIV3Schema: + description: "The TLSRoute resource is similar to TCPRoute, but can be configured + to match against TLS-specific metadata. This allows more flexibility in + matching streams for a given TLS listener. \n If you need to forward traffic + to a single target for a TLS listener, you could choose to use a TCPRoute + with a TLS listener." + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of TLSRoute. + properties: + hostnames: + description: "Hostnames defines a set of SNI names that should match + against the SNI attribute of TLS ClientHello message in TLS handshake. + This matches the RFC 1123 definition of a hostname with 2 notable + exceptions: \n 1. IPs are not allowed in SNI names per RFC 6066. + 2. A hostname may be prefixed with a wildcard label (`*.`). The + wildcard label must appear by itself as the first label. \n If a + hostname is specified by both the Listener and TLSRoute, there must + be at least one intersecting hostname for the TLSRoute to be attached + to the Listener. For example: \n * A Listener with `test.example.com` + as the hostname matches TLSRoutes that have either not specified + any hostnames, or have specified at least one of `test.example.com` + or `*.example.com`. * A Listener with `*.example.com` as the hostname + matches TLSRoutes that have either not specified any hostnames or + have specified at least one hostname that matches the Listener hostname. + For example, `test.example.com` and `*.example.com` would both match. + On the other hand, `example.com` and `test.example.net` would not + match. \n If both the Listener and TLSRoute have specified hostnames, + any TLSRoute hostnames that do not match the Listener hostname MUST + be ignored. For example, if a Listener specified `*.example.com`, + and the TLSRoute specified `test.example.com` and `test.example.net`, + `test.example.net` must not be considered for a match. \n If both + the Listener and TLSRoute have specified hostnames, and none match + with the criteria above, then the TLSRoute is not accepted. The + implementation must raise an 'Accepted' Condition with a status + of `False` in the corresponding RouteParentStatus. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged. \n Note that for ParentRefs that cross namespace + boundaries, there are specific rules. Cross-namespace references + are only valid if they are explicitly allowed by something in the + namespace they are referring to. For example, Gateway has the AllowedRoutes + field, and ReferenceGrant provides a generic way to enable any other + kind of cross-namespace reference." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the core + API group (such as for a \"Service\" kind referent), Group + must be explicitly set to \"\" (empty string). \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) \n Support: Implementation-specific (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified, this refers to the local namespace of the Route. + \n Note that there are specific rules for ParentRefs which + cross namespace boundaries. Cross-namespace references are + only valid if they are explicitly allowed by something in + the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides a + generic way to enable any other kind of cross-namespace reference. + \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. It + can be interpreted differently based on the type of parent + resource. \n When the parent resource is a Gateway, this targets + all listeners listening on the specified port that also support + this kind of Route(and select this Route). It's not recommended + to set `Port` unless the networking behaviors specified in + a Route must apply to a specific port as opposed to a listener(s) + whose port(s) may be changed. When both Port and SectionName + are specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY choose + to support other parent resources. Implementations supporting + other types of parent resources MUST clearly document how/if + Port is interpreted. \n For the purpose of status, an attachment + is considered successful as long as the parent resource accepts + it partially. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + description: Rules are a list of TLS matchers and actions. + items: + description: TLSRouteRule is the configuration for a given rule. + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. If unspecified or invalid (refers + to a non-existent resource or a Service with no endpoints), + the rule performs no forwarding; if no filters are specified + that would result in a response being sent, the underlying + implementation must actively reject request attempts to this + backend, by rejecting the connection or returning a 500 status + code. Request rejections must respect weight; if an invalid + backend is requested to have 80% of requests, then 80% of + requests must be rejected instead. \n Support: Core for Kubernetes + Service \n Support: Extended for Kubernetes ServiceImport + \n Support: Implementation-specific for any other resource + \n Support for weight: Extended" + items: + description: "BackendRef defines how a Route should forward + a request to a Kubernetes resource. \n Note that when a + namespace different than the local namespace is specified, + a ReferenceGrant object is required in the referent namespace + to allow that namespace's owner to accept the reference. + See the ReferenceGrant documentation for details." + properties: + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: "Kind is the Kubernetes resource kind of + the referent. For example \"Service\". \n Defaults to + \"Service\" when not specified. \n ExternalName services + can refer to CNAME DNS records that may live outside + of the cluster and as such are difficult to reason about + in terms of conformance. They also may not be safe to + forward to (see CVE-2021-25740 for more information). + Implementations SHOULD NOT support ExternalName Services. + \n Support: Core (Services with a type other than ExternalName) + \n Support: Implementation-specific (Services with type + ExternalName)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace different than the local + namespace is specified, a ReferenceGrant object is required + in the referent namespace to allow that namespace's + owner to accept the reference. See the ReferenceGrant + documentation for details. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + required: + - name + type: object + maxItems: 16 + minItems: 1 + type: array + type: object + maxItems: 16 + minItems: 1 + type: array + required: + - rules + type: object + status: + description: Status defines the current state of TLSRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, \n type FooStatus struct{ + // Represents the observations of a foo's current state. + // Known .status.conditions.type are: \"Available\", \"Progressing\", + and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge + // +listType=map // +listMapKey=type Conditions []metav1.Condition + `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" + protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields + }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: gateway.networking.k8s.io + description: "Group is the group of the referent. When unspecified, + \"gateway.networking.k8s.io\" is inferred. To set the + core API group (such as for a \"Service\" kind referent), + Group must be explicitly set to \"\" (empty string). \n + Support: Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) \n Support: Implementation-specific (Other + Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified, this refers to the local namespace of + the Route. \n Note that there are specific rules for ParentRefs + which cross namespace boundaries. Cross-namespace references + are only valid if they are explicitly allowed by something + in the namespace they are referring to. For example: Gateway + has the AllowedRoutes field, and ReferenceGrant provides + a generic way to enable any other kind of cross-namespace + reference. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: null + storedVersions: null +{{- end }} diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/authorization-policy.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/authorization-policy.yaml new file mode 100644 index 0000000000..7d86520e2e --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/authorization-policy.yaml @@ -0,0 +1,99 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: authorizationpolicies.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + scope: Namespaced + names: + kind: AuthorizationPolicy + plural: authorizationpolicies + singular: authorizationpolicy + shortNames: [authzpolicy] + versions: + - name: v1alpha1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + description: >- + Authorizes clients to communicate with Linkerd-proxied server + resources. + type: object + required: [targetRef, requiredAuthenticationRefs] + properties: + targetRef: + description: >- + TargetRef references a resource to which the authorization + policy applies. + type: object + required: [kind, name] + # Modified from the gateway API. + # Copyright 2020 The Kubernetes Authors + properties: + group: + description: >- + Group is the group of the referent. When empty, the + Kubernetes core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: >- + Kind is the kind of the referent. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + requiredAuthenticationRefs: + description: >- + RequiredAuthenticationRefs enumerates a set of required + authentications. ALL authentications must be satisfied for + the authorization to apply. If any of the referred objects + cannot be found, the authorization will be ignored. + type: array + items: + type: object + required: [kind, name] + properties: + group: + description: >- + Group is the group of the referent. When empty, the + Kubernetes core API group is inferred." + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: >- + Kind is the kind of the referent. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: >- + Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: >- + Name is the name of the referent. When unspecified, + this authentication refers to the local namespace. + maxLength: 253 + type: string diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/egress-network.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/egress-network.yaml new file mode 100644 index 0000000000..4289de0577 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/egress-network.yaml @@ -0,0 +1,123 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: egressnetworks.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + names: + categories: + - policy + kind: EgressNetwork + listKind: EgressNetworkList + plural: egressnetworks + singular: egressnetwork + scope: Namespaced + versions: + - name: v1alpha1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + description: >- + An EgressNetwork captures traffic to egress destinations + type: object + required: [spec] + properties: + apiVerson: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + trafficPolicy: + description: >- + This field controls the traffic policy enforced upon traffic + that does not match any explicit route resources associated + with an instance of this object. The values that are allowed + currently are: + - Allow - permits all traffic, even if it has not been + explicitly described via attaching an xRoute + resources. + - Deny - blocks all traffic that has not been described via + attaching an xRoute resource. + type: string + enum: + - Allow + - Deny + networks: + type: array + items: + type: object + required: [cidr] + properties: + cidr: + description: >- + The CIDR of the network to be authorized. + type: string + except: + description: >- + A list of IP networks/addresses not to be included in + the above `cidr`. + type: array + items: + type: string + type: object + required: + - trafficPolicy + status: + type: object + properties: + conditions: + type: array + items: + type: object + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the + condition transitioned from one status to another. + format: date-time + type: string + status: + description: status of the condition (one of True, False, Unknown) + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of the condition in CamelCase or in + foo.example.com/CamelCase. + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last + transition. Producers of specific condition types may + define expected values and meanings for this field, and + whether the values are considered a guaranteed API. The + value should be a CamelCase string. This field may not + be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + message: + description: message is a human readable message + indicating details about the transition. This may be an + empty string. + maxLength: 32768 + type: string + required: + - status + - type diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/http-local-ratelimit-policy.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/http-local-ratelimit-policy.yaml new file mode 100644 index 0000000000..b539549459 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/http-local-ratelimit-policy.yaml @@ -0,0 +1,215 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: httplocalratelimitpolicies.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + names: + kind: HTTPLocalRateLimitPolicy + listKind: HTTPLocalRateLimitPolicyList + plural: httplocalratelimitpolicies + singular: httplocalratelimitpolicy + scope: Namespaced + versions: + - name: v1alpha1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + type: object + required: [targetRef] + properties: + targetRef: + description: >- + TargetRef references a resource to which the rate limit + policy applies. Only Server is allowed. + type: object + required: [kind, name] + properties: + group: + description: >- + Group is the group of the referent. When empty, the + Kubernetes core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is the kind of the referent. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + total: + description: >- + Overall rate-limit, which all traffic coming to this + target should abide. + If unset no overall limit is applied. + type: object + required: [requestsPerSecond] + properties: + requestsPerSecond: + format: int64 + type: integer + identity: + description: >- + Fairness for individual identities; each separate client, + grouped by identity, will have this rate-limit. The + requestsPerSecond value should be less than or equal to the + total requestsPerSecond (if set). + type: object + required: [requestsPerSecond] + properties: + requestsPerSecond: + format: int64 + type: integer + overrides: + description: >- + Overrides for traffic from a specific client. The + requestsPerSecond value should be less than or equal to the + total requestsPerSecond (if set). + type: array + items: + type: object + required: [requestsPerSecond, clientRefs] + properties: + requestsPerSecond: + format: int64 + type: integer + clientRefs: + type: array + items: + type: object + required: [kind, name] + properties: + group: + description: >- + Group is the group of the referent. When empty, the + Kubernetes core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: Kind is the kind of the referent. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + namespace: + description: >- + Namespace is the namespace of the referent. + When unspecified (or empty string), this refers to the + local namespace of the Policy. + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + status: + type: object + properties: + conditions: + type: array + items: + type: object + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the + condition transitioned from one status to another. + format: date-time + type: string + status: + description: status of the condition (one of True, False, Unknown) + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of the condition in CamelCase or in + foo.example.com/CamelCase. + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last + transition. Producers of specific condition types may + define expected values and meanings for this field, and + whether the values are considered a guaranteed API. The + value should be a CamelCase string. This field may not + be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + message: + description: message is a human readable message + indicating details about the transition. This may be an + empty string. + maxLength: 32768 + type: string + required: + - status + - type + targetRef: + properties: + group: + default: policy.linkerd.io + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Server + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + maxLength: 253 + minLength: 1 + type: string + required: + - name + type: object + required: + - targetRef + additionalPrinterColumns: + - name: Target_kind + description: The resource kind to which the rate-limit applies + type: string + jsonPath: .spec.targetRef.kind + - name: Target_name + type: string + description: The resource name to which the rate-limit applies + jsonPath: .spec.targetRef.name + - name: Total_RPS + description: The overall rate-limit + type: integer + format: int32 + jsonPath: .spec.total.requestsPerSecond + - name: Identity_RPS + description: The rate-limit per identity + type: integer + format: int32 + jsonPath: .spec.identity.requestsPerSecond diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/httproute.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/httproute.yaml new file mode 100644 index 0000000000..6d2e8b07ef --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/httproute.yaml @@ -0,0 +1,5328 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: httproutes.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + names: + kind: HTTPRoute + listKind: HTTPRouteList + plural: httproutes + singular: httproute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute to process the + request. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname may + be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + \ that have either not specified any hostnames or have specified + at least one hostname that matches the Listener hostname. For + example, `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified (or empty string), this refers to the local namespace + of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "port" + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches) and processing it (filters). + properties: + backendRefs: + type: array + items: + type: object + properties: + name: + type: string + port: + type: integer + namespace: + type: string + default: "default" + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or custom conformance. + \n All filters are expected to be compatible with each other + except for the URLRewrite and RequestRedirect filters, which + may not be combined. If an implementation can not support + other combinations of filters, they must clearly document + that limitation. In all cases where incompatible or unsupported + filters are specified, implementations MUST add a warning + condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: add: - name: \"my-header\" value: + \"bar\" \n Output: GET /foo HTTP/1.1 my-header: + foo my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + \ my-header2: bar my-header3: baz \n Config: + \ remove: [\"my-header1\", \"my-header3\"] \n Output: + \ GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + \ set: - name: \"my-header\" value: \"bar\" + \n Output: GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. When empty, + port (if specified) of the request is used. \n Support: + Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\". + All implementations must support core filters. \n\n " + enum: + - RequestHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - + name: \"version\" value: \"v2\" - path: value: \"/v2/foo\" + ``` \n For a request to match against this rule, a request + must satisfy EITHER of the two conditions: \n - path prefixed + with `/foo` AND contains the header `version: v2` - path prefix + of `/v2/foo` \n See the documentation for HTTPRouteMatch on + how to specify multiple match conditions that should be ANDed + together. \n If no matches are specified, the default is a + prefix path match on \"/\", which has the effect of matching + every HTTP request. \n Proxy or Load Balancer routing configuration + generated from HTTPRoutes MUST prioritize rules based on the + following criteria, continuing on ties. Precedence must be + given to the the Rule with the largest number of: \n * Characters + in a matching non-wildcard hostname. * Characters in a matching + hostname. * Characters in a matching path. * Header matches. + * Query param matches. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within the Route that has been given precedence, + matching precedence MUST be granted to the first matching + rule meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: path: value: \"/foo\" headers: - name: + \"version\" value \"v1\" ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Custom (RegularExpression) \n Since + RegularExpression HeaderMatchType has custom conformance, + implementations can support POSIX, PCRE or any + other dialects of regular expressions. Please + read the implementation's documentation to determine + the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Custom (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + maxLength: 256 + minLength: 1 + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Custom (RegularExpression) + \n Since RegularExpression QueryParamMatchType + has custom conformance, implementations can support + POSIX, PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, type FooStatus struct{ + \ // Represents the observations of a foo's current state. + \ // Known .status.conditions.type are: \"Available\", + \"Progressing\", and \"Degraded\" // +patchMergeKey=type + \ // +patchStrategy=merge // +listType=map // + +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" + patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` + \n // other fields }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified (or empty string), this refers to the + local namespace of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta1 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute to process the + request. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname may + be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + \ that have either not specified any hostnames or have specified + at least one hostname that matches the Listener hostname. For + example, `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified (or empty string), this refers to the local namespace + of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "port" + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches) and processing it (filters). + properties: + backendRefs: + type: array + items: + type: object + properties: + name: + type: string + port: + type: integer + namespace: + type: string + default: "default" + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or custom conformance. + \n All filters are expected to be compatible with each other + except for the URLRewrite and RequestRedirect filters, which + may not be combined. If an implementation can not support + other combinations of filters, they must clearly document + that limitation. In all cases where incompatible or unsupported + filters are specified, implementations MUST add a warning + condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: add: - name: \"my-header\" value: + \"bar\" \n Output: GET /foo HTTP/1.1 my-header: + foo my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + \ my-header2: bar my-header3: baz \n Config: + \ remove: [\"my-header1\", \"my-header3\"] \n Output: + \ GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + \ set: - name: \"my-header\" value: \"bar\" + \n Output: GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. When empty, + port (if specified) of the request is used. \n Support: + Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\"." + enum: + - RequestHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - + name: \"version\" value: \"v2\" - path: value: \"/v2/foo\" + ``` \n For a request to match against this rule, a request + must satisfy EITHER of the two conditions: \n - path prefixed + with `/foo` AND contains the header `version: v2` - path prefix + of `/v2/foo` \n See the documentation for HTTPRouteMatch on + how to specify multiple match conditions that should be ANDed + together. \n If no matches are specified, the default is a + prefix path match on \"/\", which has the effect of matching + every HTTP request. \n Proxy or Load Balancer routing configuration + generated from HTTPRoutes MUST prioritize rules based on the + following criteria, continuing on ties. Precedence must be + given to the the Rule with the largest number of: \n * Characters + in a matching non-wildcard hostname. * Characters in a matching + hostname. * Characters in a matching path. * Header matches. + * Query param matches. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within the Route that has been given precedence, + matching precedence MUST be granted to the first matching + rule meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: path: value: \"/foo\" headers: - name: + \"version\" value \"v1\" ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Custom (RegularExpression) \n Since + RegularExpression HeaderMatchType has custom conformance, + implementations can support POSIX, PCRE or any + other dialects of regular expressions. Please + read the implementation's documentation to determine + the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Custom (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + maxLength: 256 + minLength: 1 + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Custom (RegularExpression) + \n Since RegularExpression QueryParamMatchType + has custom conformance, implementations can support + POSIX, PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, type FooStatus struct{ + \ // Represents the observations of a foo's current state. + \ // Known .status.conditions.type are: \"Available\", + \"Progressing\", and \"Degraded\" // +patchMergeKey=type + \ // +patchStrategy=merge // +listType=map // + +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" + patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` + \n // other fields }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified (or empty string), this refers to the + local namespace of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta2 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute to process the + request. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname may + be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + \ that have either not specified any hostnames or have specified + at least one hostname that matches the Listener hostname. For + example, `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified (or empty string), this refers to the local namespace + of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port specifies the destination + port number to use for this resource. + Port is required when the referent is + a Kubernetes Service. In this case, the + port number is the service port number, + not the target port. For other resources, + destination port might be derived from + the referent resource or this field. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches) and processing it (filters). + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. \n Failure behavior here depends + on how many BackendRefs are specified and how many are invalid. + \n If *all* entries in BackendRefs are invalid, and there + are also no filters specified in this route rule, *all* traffic + which matches this rule MUST receive a 500 status code. \n + See the HTTPBackendRef definition for the rules about what + makes a single HTTPBackendRef invalid. \n When a HTTPBackendRef + is invalid, 500 status codes MUST be returned for requests + that would have otherwise been routed to an invalid backend. + If multiple backends are specified, and some are invalid, + the proportion of requests that would otherwise have been + routed to an invalid backend MUST receive a 500 status code. + \n For example, if two backends are specified with equal weights, + and one is invalid, 50 percent of traffic must receive a 500. + Implementations may choose how that 50 percent is determined. + \n Support: Core for Kubernetes Service \n Support: Implementation-specific + for any other resource \n Support for weight: Core" + items: + description: HTTPBackendRef defines how a HTTPRoute should + forward an HTTP request. + properties: + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: Kind is kind of the referent. For example + "HTTPRoute" or "Service". Defaults to "Service" when + not specified. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace is specified, a ReferenceGrant + object is required in the referent namespace to allow + that namespace's owner to accept the reference. See + the ReferenceGrant documentation for details. \n Support: + Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + required: + - name + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or custom conformance. + \n All filters are expected to be compatible with each other + except for the URLRewrite and RequestRedirect filters, which + may not be combined. If an implementation can not support + other combinations of filters, they must clearly document + that limitation. In all cases where incompatible or unsupported + filters are specified, implementations MUST add a warning + condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: add: - name: \"my-header\" value: + \"bar\" \n Output: GET /foo HTTP/1.1 my-header: + foo my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + \ my-header2: bar my-header3: baz \n Config: + \ remove: [\"my-header1\", \"my-header3\"] \n Output: + \ GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + \ set: - name: \"my-header\" value: \"bar\" + \n Output: GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. When empty, + port (if specified) of the request is used. \n Support: + Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\"." + enum: + - RequestHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - + name: \"version\" value: \"v2\" - path: value: \"/v2/foo\" + ``` \n For a request to match against this rule, a request + must satisfy EITHER of the two conditions: \n - path prefixed + with `/foo` AND contains the header `version: v2` - path prefix + of `/v2/foo` \n See the documentation for HTTPRouteMatch on + how to specify multiple match conditions that should be ANDed + together. \n If no matches are specified, the default is a + prefix path match on \"/\", which has the effect of matching + every HTTP request. \n Proxy or Load Balancer routing configuration + generated from HTTPRoutes MUST prioritize rules based on the + following criteria, continuing on ties. Precedence must be + given to the the Rule with the largest number of: \n * Characters + in a matching non-wildcard hostname. * Characters in a matching + hostname. * Characters in a matching path. * Header matches. + * Query param matches. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within the Route that has been given precedence, + matching precedence MUST be granted to the first matching + rule meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: path: value: \"/foo\" headers: - name: + \"version\" value \"v1\" ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Custom (RegularExpression) \n Since + RegularExpression HeaderMatchType has custom conformance, + implementations can support POSIX, PCRE or any + other dialects of regular expressions. Please + read the implementation's documentation to determine + the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Custom (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + maxLength: 256 + minLength: 1 + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Custom (RegularExpression) + \n Since RegularExpression QueryParamMatchType + has custom conformance, implementations can support + POSIX, PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, type FooStatus struct{ + \ // Represents the observations of a foo's current state. + \ // Known .status.conditions.type are: \"Available\", + \"Progressing\", and \"Degraded\" // +patchMergeKey=type + \ // +patchStrategy=merge // +listType=map // + +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" + patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` + \n // other fields }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified (or empty string), this refers to the + local namespace of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .spec.hostnames + name: Hostnames + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1beta3 + schema: + openAPIV3Schema: + description: HTTPRoute provides a way to route HTTP requests. This includes + the capability to match requests by hostname, path, header, or query param. + Filters can be used to specify additional processing steps. Backends specify + where matching requests should be routed. + properties: + apiVersion: + description: 'APIVersion defines the versioned schema of this representation + of an object. Servers should convert recognized schemas to the latest + internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources' + type: string + kind: + description: 'Kind is a string value representing the REST resource this + object represents. Servers may infer this from the endpoint the client + submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + metadata: + type: object + spec: + description: Spec defines the desired state of HTTPRoute. + properties: + hostnames: + description: "Hostnames defines a set of hostname that should match + against the HTTP Host header to select a HTTPRoute to process the + request. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname may + be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n If a hostname is specified + by both the Listener and HTTPRoute, there must be at least one intersecting + hostname for the HTTPRoute to be attached to the Listener. For example: + \n * A Listener with `test.example.com` as the hostname matches + HTTPRoutes that have either not specified any hostnames, or have + specified at least one of `test.example.com` or `*.example.com`. + * A Listener with `*.example.com` as the hostname matches HTTPRoutes + \ that have either not specified any hostnames or have specified + at least one hostname that matches the Listener hostname. For + example, `*.example.com`, `test.example.com`, and `foo.test.example.com` + would all match. On the other hand, `example.com` and `test.example.net` + would not match. \n Hostnames that are prefixed with a wildcard + label (`*.`) are interpreted as a suffix match. That means that + a match for `*.example.com` would match both `test.example.com`, + and `foo.test.example.com`, but not `example.com`. \n If both the + Listener and HTTPRoute have specified hostnames, any HTTPRoute hostnames + that do not match the Listener hostname MUST be ignored. For example, + if a Listener specified `*.example.com`, and the HTTPRoute specified + `test.example.com` and `test.example.net`, `test.example.net` must + not be considered for a match. \n If both the Listener and HTTPRoute + have specified hostnames, and none match with the criteria above, + then the HTTPRoute is not accepted. The implementation must raise + an 'Accepted' Condition with a status of `False` in the corresponding + RouteParentStatus. \n Support: Core" + items: + description: "Hostname is the fully qualified domain name of a network + host. This matches the RFC 1123 definition of a hostname with + 2 notable exceptions: \n 1. IPs are not allowed. 2. A hostname + may be prefixed with a wildcard label (`*.`). The wildcard label + must appear by itself as the first label. \n Hostname can be \"precise\" + which is a domain name without the terminating dot of a network + host (e.g. \"foo.example.com\") or \"wildcard\", which is a domain + name prefixed with a single wildcard label (e.g. `*.example.com`). + \n Note that as per RFC1035 and RFC1123, a *label* must consist + of lower case alphanumeric characters or '-', and must start and + end with an alphanumeric character. No other punctuation is allowed." + maxLength: 253 + minLength: 1 + pattern: ^(\*\.)?[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + maxItems: 16 + type: array + parentRefs: + description: "ParentRefs references the resources (usually Gateways) + that a Route wants to be attached to. Note that the referenced parent + resource needs to allow this for the attachment to be complete. + For Gateways, that means the Gateway needs to allow attachment from + Routes of this kind and namespace. \n The only kind of parent resource + with \"Core\" support is Gateway. This API may be extended in the + future to support additional kinds of parent resources such as one + of the route kinds. \n It is invalid to reference an identical parent + more than once. It is valid to reference multiple distinct sections + within the same parent resource, such as 2 Listeners within a Gateway. + \n It is possible to separately reference multiple distinct objects + that may be collapsed by an implementation. For example, some implementations + may choose to merge compatible Gateway Listeners together. If that + is the case, the list of routes attached to those resources should + also be merged." + items: + description: "ParentReference identifies an API object (usually + a Gateway) that can be considered a parent of this resource (usually + a route). The only kind of parent resource with \"Core\" support + is Gateway. This API may be extended in the future to support + additional kinds of parent resources, such as HTTPRoute. \n The + API object must be valid in the cluster; the Group and Kind must + be registered in the cluster for this reference to be valid." + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: Core + (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. When + unspecified (or empty string), this refers to the local namespace + of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port specifies the destination + port number to use for this resource. + Port is required when the referent is + a Kubernetes Service. In this case, the + port number is the service port number, + not the target port. For other resources, + destination port might be derived from + the referent resource or this field. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within the + target resource. In the following resources, SectionName is + interpreted as the following: \n * Gateway: Listener Name. + When both Port (experimental) and SectionName are specified, + the name and port of the selected listener must match both + specified values. \n Implementations MAY choose to support + attaching Routes to other resources. If that is the case, + they MUST clearly document how SectionName is interpreted. + \n When unspecified (empty string), this will reference the + entire resource. For the purpose of status, an attachment + is considered successful if at least one section in the parent + resource accepts it. For example, Gateway listeners can restrict + which Routes can attach to them by Route kind, namespace, + or hostname. If 1 of 2 Gateway listeners accept attachment + from the referencing Route, the Route MUST be considered successfully + attached. If no Gateway listeners accept attachment from this + Route, the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + maxItems: 32 + type: array + rules: + default: + - matches: + - path: + type: PathPrefix + value: / + description: Rules are a list of HTTP matchers, filters and actions. + items: + description: HTTPRouteRule defines semantics for matching an HTTP + request based on conditions (matches) and processing it (filters). + properties: + backendRefs: + description: "BackendRefs defines the backend(s) where matching + requests should be sent. \n Failure behavior here depends + on how many BackendRefs are specified and how many are invalid. + \n If *all* entries in BackendRefs are invalid, and there + are also no filters specified in this route rule, *all* traffic + which matches this rule MUST receive a 500 status code. \n + See the HTTPBackendRef definition for the rules about what + makes a single HTTPBackendRef invalid. \n When a HTTPBackendRef + is invalid, 500 status codes MUST be returned for requests + that would have otherwise been routed to an invalid backend. + If multiple backends are specified, and some are invalid, + the proportion of requests that would otherwise have been + routed to an invalid backend MUST receive a 500 status code. + \n For example, if two backends are specified with equal weights, + and one is invalid, 50 percent of traffic must receive a 500. + Implementations may choose how that 50 percent is determined. + \n Support: Core for Kubernetes Service \n Support: Implementation-specific + for any other resource \n Support for weight: Core" + items: + description: HTTPBackendRef defines how a HTTPRoute should + forward an HTTP request. + properties: + group: + default: "" + description: Group is the group of the referent. For example, + "gateway.networking.k8s.io". When unspecified or empty + string, core API group is inferred. + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Service + description: Kind is kind of the referent. For example + "HTTPRoute" or "Service". Defaults to "Service" when + not specified. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: Name is the name of the referent. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the backend. + When unspecified, the local namespace is inferred. \n + Note that when a namespace is specified, a ReferenceGrant + object is required in the referent namespace to allow + that namespace's owner to accept the reference. See + the ReferenceGrant documentation for details. \n Support: + Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: Port specifies the destination port number + to use for this resource. Port is required when the + referent is a Kubernetes Service. In this case, the + port number is the service port number, not the target + port. For other resources, destination port might be + derived from the referent resource or this field. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + weight: + default: 1 + description: "Weight specifies the proportion of requests + forwarded to the referenced backend. This is computed + as weight/(sum of all weights in this BackendRefs list). + For non-zero values, there may be some epsilon from + the exact proportion defined here depending on the precision + an implementation supports. Weight is not a percentage + and the sum of weights does not need to equal 100. \n + If only one backend is specified and it has a weight + greater than 0, 100% of the traffic is forwarded to + that backend. If weight is set to 0, no traffic should + be forwarded for this entry. If unspecified, weight + defaults to 1. \n Support for this field varies based + on the context where used." + format: int32 + maximum: 1000000 + minimum: 0 + type: integer + filters: + description: "Filters defined at this level should be + executed if and only if the request is being forwarded + to the backend defined here. \n Support: Implementation-specific + (For broader support of filters, use the Filters field + in HTTPRouteRule.)" + items: + description: HTTPRouteFilter defines processing steps + that must be completed during the request or response + lifecycle. HTTPRouteFilters are meant as an extension + point to express processing that may be done in Gateway + implementations. Some examples include request or + response modification, implementing authentication + strategies, rate-limiting, and traffic shaping. API + guarantee/conformance is defined based on the type + of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema + for a filter that modifies request headers. \n + Support: Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for + a filter that responds to the request with an + HTTP redirection. \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be + used in the value of the `Location` header + in the response. When empty, the hostname + in the `Host` header of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in + the value of the `Location` header in the + response. \n If no port is specified, the + redirect port MUST be derived using the following + rules: \n * If redirect scheme is not-empty, + the redirect port MUST be the well-known port + associated with the redirect scheme. Specifically + \"http\" to port 80 and \"https\" to port + 443. If the redirect scheme does not have + a well-known port, the listener port of the + Gateway SHOULD be used. * If redirect scheme + is empty, the redirect port MUST be the Gateway + Listener port. \n Implementations SHOULD NOT + add the port number in the 'Location' header + in the following cases: \n * A Location header + that will use HTTP (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 80. * A Location header that + will use HTTPS (whether that is determined + via the Listener protocol or the Scheme field) + _and_ use port 443. \n Support: Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used + in the value of the `Location` header in the + response. When empty, the scheme of the request + is used. \n Scheme redirects can affect the + port of the redirect, for more information, + refer to the documentation for the port field + of this filter. \n Note that values may be + added to this enum, implementations must ensure + that unknown values will not cause a crash. + \n Unknown values here must result in the + implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`. \n Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status + code to be used in response. \n Note that + values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result + in the implementation setting the Accepted + Condition for the Route to `status: False`, + with a Reason of `UnsupportedValue`. \n Support: + Core" + enum: + - 301 + - 302 + type: integer + type: object + responseHeaderModifier: + description: "ResponseHeaderModifier defines a schema + for a filter that modifies response headers. \n + Support: Extended" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It + appends to any existing values associated + with the header name. \n Input: GET /foo HTTP/1.1 + my-header: foo \n Config: add: - name: \"my-header\" + value: \"bar,baz\" \n Output: GET /foo HTTP/1.1 + my-header: foo,bar,baz" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from + the HTTP request before the action. The value + of Remove is a list of HTTP header names. + Note that the header names are case-insensitive + (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + my-header2: bar my-header3: baz \n Config: + remove: [\"my-header1\", \"my-header3\"] \n + Output: GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with + the given header (name, value) before the + action. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: set: - name: \"my-header\" + value: \"bar\" \n Output: GET /foo HTTP/1.1 + my-header: bar" + items: + description: HTTPHeader represents an HTTP + Header name and value as defined by RFC + 7230. + properties: + name: + description: "Name is the name of the + HTTP Header to be matched. Name matching + MUST be case insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an + equivalent name MUST be considered for + a match. Subsequent entries with an + equivalent header name MUST be ignored. + Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP + Header to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: + description: "Type identifies the type of filter + to apply. As with other API fields, types are + classified into three conformance levels: \n - + Core: Filter types and their corresponding configuration + defined by \"Support: Core\" in this package, + e.g. \"RequestHeaderModifier\". All implementations + must support core filters. \n - Extended: Filter + types and their corresponding configuration defined + by \"Support: Extended\" in this package, e.g. + \"RequestMirror\". Implementers are encouraged + to support extended filters. \n - Implementation-specific: + Filters that are defined and supported by specific + vendors. In the future, filters showing convergence + in behavior across multiple implementations will + be considered for inclusion in extended or core + conformance levels. Filter-specific configuration + for such filters is specified using the ExtensionRef + field. `Type` should be set to \"ExtensionRef\" + for custom filters. \n Implementers are encouraged + to define custom implementation types to extend + the core API with implementation-specific behavior. + \n If a reference to a custom filter type cannot + be resolved, the filter MUST NOT be skipped. Instead, + requests that would have been processed by that + filter MUST receive a HTTP error response. \n + Note that values may be added to this enum, implementations + must ensure that unknown values will not cause + a crash. \n Unknown values here must result in + the implementation setting the Accepted Condition + for the Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + required: + - name + type: object + maxItems: 16 + type: array + filters: + description: "Filters define the filters that are applied to + requests that match this rule. \n The effects of ordering + of multiple behaviors are currently unspecified. This can + change in the future based on feedback during the alpha stage. + \n Conformance-levels at this level are defined based on the + type of filter: \n - ALL core filters MUST be supported by + all implementations. - Implementers are encouraged to support + extended filters. - Implementation-specific custom filters + have no API guarantees across implementations. \n Specifying + a core filter multiple times has unspecified or custom conformance. + \n All filters are expected to be compatible with each other + except for the URLRewrite and RequestRedirect filters, which + may not be combined. If an implementation can not support + other combinations of filters, they must clearly document + that limitation. In all cases where incompatible or unsupported + filters are specified, implementations MUST add a warning + condition to status. \n Support: Core" + items: + description: HTTPRouteFilter defines processing steps that + must be completed during the request or response lifecycle. + HTTPRouteFilters are meant as an extension point to express + processing that may be done in Gateway implementations. + Some examples include request or response modification, + implementing authentication strategies, rate-limiting, and + traffic shaping. API guarantee/conformance is defined based + on the type of the filter. + properties: + requestHeaderModifier: + description: "RequestHeaderModifier defines a schema for + a filter that modifies request headers. \n Support: + Core" + properties: + add: + description: "Add adds the given header(s) (name, + value) to the request before the action. It appends + to any existing values associated with the header + name. \n Input: GET /foo HTTP/1.1 my-header: + foo \n Config: add: - name: \"my-header\" value: + \"bar\" \n Output: GET /foo HTTP/1.1 my-header: + foo my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + description: "Remove the given header(s) from the + HTTP request before the action. The value of Remove + is a list of HTTP header names. Note that the header + names are case-insensitive (see https://datatracker.ietf.org/doc/html/rfc2616#section-4.2). + \n Input: GET /foo HTTP/1.1 my-header1: foo + \ my-header2: bar my-header3: baz \n Config: + \ remove: [\"my-header1\", \"my-header3\"] \n Output: + \ GET /foo HTTP/1.1 my-header2: bar" + items: + type: string + maxItems: 16 + type: array + set: + description: "Set overwrites the request with the + given header (name, value) before the action. \n + Input: GET /foo HTTP/1.1 my-header: foo \n Config: + \ set: - name: \"my-header\" value: \"bar\" + \n Output: GET /foo HTTP/1.1 my-header: bar" + items: + description: HTTPHeader represents an HTTP Header + name and value as defined by RFC 7230. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case + insensitive. (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent + header names, the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST + be ignored. Due to the case-insensitivity + of header names, \"foo\" and \"Foo\" are considered + equivalent." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + description: Value is the value of HTTP Header + to be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + requestRedirect: + description: "RequestRedirect defines a schema for a filter + that responds to the request with an HTTP redirection. + \n Support: Core" + properties: + hostname: + description: "Hostname is the hostname to be used + in the value of the `Location` header in the response. + When empty, the hostname of the request is used. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + path: + description: "Path defines parameters used to + modify the path of the incoming request. The + modified path is then used to construct the + `Location` header. When empty, the request + path is used as-is. \n Support: Extended" + properties: + replaceFullPath: + description: ReplaceFullPath specifies the + value with which to replace the full path + of a request during a rewrite or redirect. + maxLength: 1024 + type: string + replacePrefixMatch: + description: "ReplacePrefixMatch specifies + the value with which to replace the prefix + match of a request during a rewrite or + redirect. For example, a request to \"/foo/bar\" + with a prefix match of \"/foo\" and a + ReplacePrefixMatch of \"/xyz\" would be + modified to \"/xyz/bar\". \n Note that + this matches the behavior of the PathPrefix + match type. This matches full path elements. + A path element refers to the list of labels + in the path split by the `/` separator. + When specified, a trailing `/` is ignored. + For example, the paths `/abc`, `/abc/`, + and `/abc/def` would all match the prefix + `/abc`, but the path `/abcd` would not. + \n Request Path | Prefix Match | Replace + Prefix | Modified Path -------------|--------------|----------------|---------- + /foo/bar | /foo | /xyz | + /xyz/bar /foo/bar | /foo | + /xyz/ | /xyz/bar /foo/bar | + /foo/ | /xyz | /xyz/bar + /foo/bar | /foo/ | /xyz/ | + /xyz/bar /foo | /foo | + /xyz | /xyz /foo/ | /foo + \ | /xyz | /xyz/ /foo/bar + \ | /foo | | + /bar /foo/ | /foo | | / /foo | /foo | + | / /foo/ | /foo + \ | / | / /foo | + /foo | / | /" + maxLength: 1024 + type: string + type: + description: "Type defines the type of path + modifier. Additional types may be added + in a future release of the API. \n Note + that values may be added to this enum, + implementations must ensure that unknown + values will not cause a crash. \n Unknown + values here must result in the implementation + setting the Accepted Condition for the + Route to `status: False`, with a Reason + of `UnsupportedValue`." + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + port: + description: "Port is the port to be used in the value + of the `Location` header in the response. When empty, + port (if specified) of the request is used. \n Support: + Extended" + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + description: "Scheme is the scheme to be used in the + value of the `Location` header in the response. + When empty, the scheme of the request is used. \n + Support: Extended" + enum: + - http + - https + type: string + statusCode: + default: 302 + description: "StatusCode is the HTTP status code to + be used in response. \n Support: Core" + enum: + - 301 + - 302 + type: integer + type: object + type: + description: "Type identifies the type of filter to apply. + As with other API fields, types are classified into + three conformance levels: \n - Core: Filter types and + their corresponding configuration defined by \"Support: + Core\" in this package, e.g. \"RequestHeaderModifier\"." + enum: + - RequestHeaderModifier + - RequestRedirect + type: string + required: + - type + type: object + maxItems: 16 + type: array + matches: + default: + - path: + type: PathPrefix + value: / + description: "Matches define conditions used for matching the + rule against incoming HTTP requests. Each match is independent, + i.e. this rule will be matched if **any** one of the matches + is satisfied. \n For example, take the following matches configuration: + \n ``` matches: - path: value: \"/foo\" headers: - + name: \"version\" value: \"v2\" - path: value: \"/v2/foo\" + ``` \n For a request to match against this rule, a request + must satisfy EITHER of the two conditions: \n - path prefixed + with `/foo` AND contains the header `version: v2` - path prefix + of `/v2/foo` \n See the documentation for HTTPRouteMatch on + how to specify multiple match conditions that should be ANDed + together. \n If no matches are specified, the default is a + prefix path match on \"/\", which has the effect of matching + every HTTP request. \n Proxy or Load Balancer routing configuration + generated from HTTPRoutes MUST prioritize rules based on the + following criteria, continuing on ties. Precedence must be + given to the the Rule with the largest number of: \n * Characters + in a matching non-wildcard hostname. * Characters in a matching + hostname. * Characters in a matching path. * Header matches. + * Query param matches. \n If ties still exist across multiple + Routes, matching precedence MUST be determined in order of + the following criteria, continuing on ties: \n * The oldest + Route based on creation timestamp. * The Route appearing first + in alphabetical order by \"{namespace}/{name}\". \n If ties + still exist within the Route that has been given precedence, + matching precedence MUST be granted to the first matching + rule meeting the above criteria. \n When no rules matching + a request have been successfully attached to the parent a + request is coming from, a HTTP 404 status code MUST be returned." + items: + description: "HTTPRouteMatch defines the predicate used to + match requests to a given action. Multiple match types are + ANDed together, i.e. the match will evaluate to true only + if all conditions are satisfied. \n For example, the match + below will match a HTTP request only if its path starts + with `/foo` AND it contains the `version: v1` header: \n + ``` match: path: value: \"/foo\" headers: - name: + \"version\" value \"v1\" ```" + properties: + headers: + description: Headers specifies HTTP request header matchers. + Multiple match values are ANDed together, meaning, a + request must match all the specified headers to select + the route. + items: + description: HTTPHeaderMatch describes how to select + a HTTP route by matching HTTP request headers. + properties: + name: + description: "Name is the name of the HTTP Header + to be matched. Name matching MUST be case insensitive. + (See https://tools.ietf.org/html/rfc7230#section-3.2). + \n If multiple entries specify equivalent header + names, only the first entry with an equivalent + name MUST be considered for a match. Subsequent + entries with an equivalent header name MUST be + ignored. Due to the case-insensitivity of header + names, \"foo\" and \"Foo\" are considered equivalent. + \n When a header is repeated in an HTTP request, + it is implementation-specific behavior as to how + this is represented. Generally, proxies should + follow the guidance from the RFC: https://www.rfc-editor.org/rfc/rfc7230.html#section-3.2.2 + regarding processing a repeated header, with special + handling for \"Set-Cookie\"." + maxLength: 256 + minLength: 1 + pattern: ^[A-Za-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the header. \n Support: Core (Exact) + \n Support: Custom (RegularExpression) \n Since + RegularExpression HeaderMatchType has custom conformance, + implementations can support POSIX, PCRE or any + other dialects of regular expressions. Please + read the implementation's documentation to determine + the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP Header to + be matched. + maxLength: 4096 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + method: + description: "Method specifies HTTP method matcher. When + specified, this route will be matched only if the request + has the specified method. \n Support: Extended" + enum: + - GET + - HEAD + - POST + - PUT + - DELETE + - CONNECT + - OPTIONS + - TRACE + - PATCH + type: string + path: + default: + type: PathPrefix + value: / + description: Path specifies a HTTP request path matcher. + If this field is not specified, a default prefix match + on the "/" path is provided. + properties: + type: + default: PathPrefix + description: "Type specifies how to match against + the path Value. \n Support: Core (Exact, PathPrefix) + \n Support: Custom (RegularExpression)" + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + default: / + description: Value of the HTTP path to match against. + maxLength: 1024 + type: string + type: object + queryParams: + description: QueryParams specifies HTTP query parameter + matchers. Multiple match values are ANDed together, + meaning, a request must match all the specified query + parameters to select the route. + items: + description: HTTPQueryParamMatch describes how to select + a HTTP route by matching HTTP query parameters. + properties: + name: + description: Name is the name of the HTTP query + param to be matched. This must be an exact string + match. (See https://tools.ietf.org/html/rfc7230#section-2.7.3). + maxLength: 256 + minLength: 1 + type: string + type: + default: Exact + description: "Type specifies how to match against + the value of the query parameter. \n Support: + Extended (Exact) \n Support: Custom (RegularExpression) + \n Since RegularExpression QueryParamMatchType + has custom conformance, implementations can support + POSIX, PCRE or any other dialects of regular expressions. + Please read the implementation's documentation + to determine the supported dialect." + enum: + - Exact + - RegularExpression + type: string + value: + description: Value is the value of HTTP query param + to be matched. + maxLength: 1024 + minLength: 1 + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + maxItems: 8 + type: array + timeouts: + description: "Timeouts defines the timeouts that can be configured + for an HTTP request. \n Support: Core \n " + properties: + backendRequest: + description: "BackendRequest specifies a timeout for an + individual request from the gateway to a backend service. + Typically used in conjunction with automatic retries, + if supported by an implementation. Default is the value + of Request timeout. \n Support: Extended" + format: duration + type: string + request: + description: "Request specifies a timeout for responding + to client HTTP requests, disabled by default. \n For example, + the following rule will timeout if a client request is + taking longer than 10 seconds to complete: \n ``` rules: + - timeouts: request: 10s backendRefs: ... ``` \n Support: + Core" + format: duration + type: string + type: object + type: object + maxItems: 16 + type: array + type: object + status: + description: Status defines the current state of HTTPRoute. + properties: + parents: + description: "Parents is a list of parent resources (usually Gateways) + that are associated with the route, and the status of the route + with respect to each parent. When this route attaches to a parent, + the controller that manages the parent must add an entry to this + list when the controller first sees the route and should update + the entry as appropriate when the route or gateway is modified. + \n Note that parent references that cannot be resolved by an implementation + of this API will not be added to this list. Implementations of this + API can only populate Route status for the Gateways/parent resources + they are responsible for. \n A maximum of 32 Gateways will be represented + in this list. An empty list means the route has not been attached + to any Gateway." + items: + description: RouteParentStatus describes the status of a route with + respect to an associated Parent. + properties: + conditions: + description: "Conditions describes the status of the route with + respect to the Gateway. Note that the route's availability + is also subject to the Gateway's own status conditions and + listener status. \n If the Route's ParentRef specifies an + existing Gateway that supports Routes of this kind AND that + Gateway's controller has sufficient access, then that Gateway's + controller MUST set the \"Accepted\" condition on the Route, + to indicate whether the route has been accepted or rejected + by the Gateway, and why. \n A Route MUST be considered \"Accepted\" + if at least one of the Route's rules is implemented by the + Gateway. \n There are a number of cases where the \"Accepted\" + condition may not be set due to lack of controller visibility, + that includes when: \n * The Route refers to a non-existent + parent. * The Route is of a type that the controller does + not support. * The Route is in a namespace the the controller + does not have access to." + items: + description: "Condition contains details for one aspect of + the current state of this API Resource. --- This struct + is intended for direct use as an array at the field path + .status.conditions. For example, type FooStatus struct{ + \ // Represents the observations of a foo's current state. + \ // Known .status.conditions.type are: \"Available\", + \"Progressing\", and \"Degraded\" // +patchMergeKey=type + \ // +patchStrategy=merge // +listType=map // + +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" + patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` + \n // other fields }" + properties: + lastTransitionTime: + description: lastTransitionTime is the last time the condition + transitioned from one status to another. This should + be when the underlying condition changed. If that is + not known, then using the time when the API field changed + is acceptable. + format: date-time + type: string + message: + description: message is a human readable message indicating + details about the transition. This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: observedGeneration represents the .metadata.generation + that the condition was set based upon. For instance, + if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration + is 9, the condition is out of date with respect to the + current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last transition. + Producers of specific condition types may define expected + values and meanings for this field, and whether the + values are considered a guaranteed API. The value should + be a CamelCase string. This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, + Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. + --- Many .condition.type values are consistent across + resources like Available, but because arbitrary conditions + can be useful (see .node.status.conditions), the ability + to deconflict is important. The regex it matches is + (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + maxItems: 8 + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + controllerName: + description: "ControllerName is a domain/path string that indicates + the name of the controller that wrote this status. This corresponds + with the controllerName field on GatewayClass. \n Example: + \"example.net/gateway-controller\". \n The format of this + field is DOMAIN \"/\" PATH, where DOMAIN and PATH are valid + Kubernetes names (https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names). + \n Controllers MUST populate this field when writing status. + Controllers should ensure that entries to status populated + with their ControllerName are cleaned up when they are no + longer necessary." + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*\/[A-Za-z0-9\/\-._~%!$&'()*+,;=:]+$ + type: string + parentRef: + description: ParentRef corresponds with a ParentRef in the spec + that this RouteParentStatus struct describes the status of. + properties: + group: + default: policy.linkerd.io + description: "Group is the group of the referent. \n Support: + Core" + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + default: Gateway + description: "Kind is kind of the referent. \n Support: + Core (Gateway) Support: Custom (Other Resources)" + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: "Name is the name of the referent. \n Support: + Core" + maxLength: 253 + minLength: 1 + type: string + namespace: + description: "Namespace is the namespace of the referent. + When unspecified (or empty string), this refers to the + local namespace of the Route. \n Support: Core" + maxLength: 63 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?$ + type: string + port: + description: "Port is the network port this Route targets. + It can be interpreted differently based on the type of + parent resource. \n When the parent resource is a Gateway, + this targets all listeners listening on the specified + port that also support this kind of Route(and select this + Route). It's not recommended to set `Port` unless the + networking behaviors specified in a Route must apply to + a specific port as opposed to a listener(s) whose port(s) + may be changed. When both Port and SectionName are specified, + the name and port of the selected listener must match + both specified values. \n Implementations MAY choose to + support other parent resources. Implementations supporting + other types of parent resources MUST clearly document + how/if Port is interpreted. \n For the purpose of status, + an attachment is considered successful as long as the + parent resource accepts it partially. For example, Gateway + listeners can restrict which Routes can attach to them + by Route kind, namespace, or hostname. If 1 of 2 Gateway + listeners accept attachment from the referencing Route, + the Route MUST be considered successfully attached. If + no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Extended \n " + format: int32 + maximum: 65535 + minimum: 1 + type: integer + sectionName: + description: "SectionName is the name of a section within + the target resource. In the following resources, SectionName + is interpreted as the following: \n * Gateway: Listener + Name. When both Port (experimental) and SectionName are + specified, the name and port of the selected listener + must match both specified values. \n Implementations MAY + choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName + is interpreted. \n When unspecified (empty string), this + will reference the entire resource. For the purpose of + status, an attachment is considered successful if at least + one section in the parent resource accepts it. For example, + Gateway listeners can restrict which Routes can attach + to them by Route kind, namespace, or hostname. If 1 of + 2 Gateway listeners accept attachment from the referencing + Route, the Route MUST be considered successfully attached. + If no Gateway listeners accept attachment from this Route, + the Route MUST be considered detached from the Gateway. + \n Support: Core" + maxLength: 253 + minLength: 1 + pattern: ^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + required: + - name + type: object + required: + - controllerName + - parentRef + type: object + maxItems: 32 + type: array + required: + - parents + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: [] + storedVersions: [] diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/meshtls-authentication.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/meshtls-authentication.yaml new file mode 100644 index 0000000000..58ee815f59 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/meshtls-authentication.yaml @@ -0,0 +1,87 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: meshtlsauthentications.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + scope: Namespaced + names: + kind: MeshTLSAuthentication + plural: meshtlsauthentications + singular: meshtlsauthentication + shortNames: [meshtlsauthn] + versions: + - name: v1alpha1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + description: >- + MeshTLSAuthentication defines a list of authenticated client IDs + to be referenced by an `AuthorizationPolicy`. If a client + connection has the mutually-authenticated identity that matches + ANY of the of the provided identities, the connection is + considered authenticated. + type: object + oneOf: + - required: [identities] + - required: [identityRefs] + properties: + identities: + description: >- + Authorizes clients with the provided proxy identity strings + (as provided via MTLS) + + The `*` prefix can be used to match all identities in + a domain. An identity string of `*` indicates that + all authentication clients are authorized. + type: array + minItems: 1 + items: + type: string + identityRefs: + type: array + minItems: 1 + items: + type: object + required: + - kind + properties: + group: + description: >- + Group is the group of the referent. When empty, the + Kubernetes core API group is inferred." + maxLength: 253 + pattern: ^$|^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$ + type: string + kind: + description: >- + Kind is the kind of the referent. + maxLength: 63 + minLength: 1 + pattern: ^[a-zA-Z]([-a-zA-Z0-9]*[a-zA-Z0-9])?$ + type: string + name: + description: >- + Name is the name of the referent. When unspecified, + this refers to all resources of the specified Group + and Kind in the specified namespace. + maxLength: 253 + minLength: 1 + type: string + namespace: + description: >- + Name is the name of the referent. When unspecified, + this authentication refers to the local namespace. + maxLength: 253 + type: string diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/network-authentication.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/network-authentication.yaml new file mode 100644 index 0000000000..cef15d3c40 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/network-authentication.yaml @@ -0,0 +1,53 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: networkauthentications.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + scope: Namespaced + names: + kind: NetworkAuthentication + plural: networkauthentications + singular: networkauthentication + shortNames: [netauthn, networkauthn] + versions: + - name: v1alpha1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + description: >- + NetworkAuthentication defines a list of authenticated client + networks to be referenced by an `AuthorizationPolicy`. If a + client connection originates from ANY of the of the provided + networks, the connection is considered authenticated. + type: object + required: [networks] + properties: + networks: + type: array + items: + type: object + required: [cidr] + properties: + cidr: + description: >- + The CIDR of the network to be authorized. + type: string + except: + description: >- + A list of IP networks/addresses not to be included in + the above `cidr`. + type: array + items: + type: string diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server-authorization.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server-authorization.yaml new file mode 100644 index 0000000000..33fb659002 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server-authorization.yaml @@ -0,0 +1,266 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: serverauthorizations.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + scope: Namespaced + names: + kind: ServerAuthorization + plural: serverauthorizations + singular: serverauthorization + shortNames: [saz, serverauthz, srvauthz] + versions: + - name: v1alpha1 + served: true + storage: false + deprecated: true + deprecationWarning: "policy.linkerd.io/v1alpha1 ServerAuthorization is deprecated; use policy.linkerd.io/v1beta1 ServerAuthorization" + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + description: >- + Authorizes clients to communicate with Linkerd-proxied servers. + type: object + required: [server, client] + properties: + server: + description: >- + Identifies servers in the same namespace for which this + authorization applies. + + Only one of `name` or `selector` may be specified. + type: object + oneOf: + - required: [name] + - required: [selector] + properties: + name: + description: References a `Server` instance by name + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + selector: + description: >- + A label query over servers on which this authorization applies. + type: object + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + client: + description: Describes clients authorized to access a server. + type: object + properties: + networks: + description: >- + Limits the client IP addresses to which this + authorization applies. If unset, the server chooses a + default (typically, all IPs or the cluster's pod + network). + type: array + items: + type: object + required: [cidr] + properties: + cidr: + type: string + except: + type: array + items: + type: string + unauthenticated: + description: >- + Authorizes unauthenticated clients to access a server. + type: boolean + meshTLS: + type: object + properties: + unauthenticatedTLS: + type: boolean + description: >- + Indicates that no client identity is required for + communication. + + This is mostly important for the identity + controller, which must terminate TLS connections + from clients that do not yet have a certificate. + identities: + description: >- + Authorizes clients with the provided proxy identity + strings (as provided via MTLS) + + The `*` prefix can be used to match all identities in + a domain. An identity string of `*` indicates that + all authentication clients are authorized. + type: array + items: + type: string + pattern: '^(\*|[a-z0-9]([-a-z0-9]*[a-z0-9])?)(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$' + serviceAccounts: + description: >- + Authorizes clients with the provided proxy identity + service accounts (as provided via MTLS) + type: array + items: + type: object + required: [name] + properties: + name: + description: The ServiceAccount's name. + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + description: >- + The ServiceAccount's namespace. If unset, the + authorization's namespace is used. + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + - name: v1beta1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + description: >- + Authorizes clients to communicate with Linkerd-proxied servers. + type: object + required: [server, client] + properties: + server: + description: >- + Identifies servers in the same namespace for which this + authorization applies. + + Only one of `name` or `selector` may be specified. + type: object + oneOf: + - required: [name] + - required: [selector] + properties: + name: + description: References a `Server` instance by name + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + selector: + description: >- + A label query over servers on which this authorization applies. + type: object + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + client: + description: Describes clients authorized to access a server. + type: object + properties: + networks: + description: >- + Limits the client IP addresses to which this + authorization applies. If unset, the server chooses a + default (typically, all IPs or the cluster's pod + network). + type: array + items: + type: object + required: [cidr] + properties: + cidr: + type: string + except: + type: array + items: + type: string + unauthenticated: + description: >- + Authorizes unauthenticated clients to access a server. + type: boolean + meshTLS: + type: object + properties: + unauthenticatedTLS: + type: boolean + description: >- + Indicates that no client identity is required for + communication. + + This is mostly important for the identity + controller, which must terminate TLS connections + from clients that do not yet have a certificate. + identities: + description: >- + Authorizes clients with the provided proxy identity + strings (as provided via MTLS) + + The `*` prefix can be used to match all identities in + a domain. An identity string of `*` indicates that + all authentication clients are authorized. + type: array + items: + type: string + pattern: '^(\*|[a-z0-9]([-a-z0-9]*[a-z0-9])?)(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$' + serviceAccounts: + description: >- + Authorizes clients with the provided proxy identity + service accounts (as provided via MTLS) + type: array + items: + type: object + required: [name] + properties: + name: + description: The ServiceAccount's name. + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + description: >- + The ServiceAccount's namespace. If unset, the + authorization's namespace is used. + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + additionalPrinterColumns: + - name: Server + type: string + description: The server that this grants access to + jsonPath: .spec.server.name diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server.yaml new file mode 100644 index 0000000000..3de5a34e0e --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/policy/server.yaml @@ -0,0 +1,319 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: servers.policy.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: policy.linkerd.io + names: + kind: Server + plural: servers + singular: server + shortNames: [srv] + scope: Namespaced + versions: + - name: v1alpha1 + served: true + storage: false + deprecated: true + deprecationWarning: "policy.linkerd.io/v1alpha1 Server is deprecated; use policy.linkerd.io/v1beta1 Server" + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + type: object + required: + - podSelector + - port + properties: + podSelector: + type: object + description: >- + Selects pods in the same namespace. + oneOf: + - required: [matchExpressions] + - required: [matchLabels] + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + port: + description: >- + A port name or number. Must exist in a pod spec. + x-kubernetes-int-or-string: true + proxyProtocol: + description: >- + Configures protocol discovery for inbound connections. + + Supersedes the `config.linkerd.io/opaque-ports` annotation. + type: string + default: unknown + - name: v1beta1 + served: true + storage: false + deprecated: true + deprecationWarning: "policy.linkerd.io/v1beta1 Server is deprecated; use policy.linkerd.io/v1beta3 Server" + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + type: object + required: + - podSelector + - port + properties: + podSelector: + type: object + description: >- + Selects pods in the same namespace. + + The result of matchLabels and matchExpressions are ANDed. + Selects all if empty. + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + port: + description: >- + A port name or number. Must exist in a pod spec. + x-kubernetes-int-or-string: true + proxyProtocol: + description: >- + Configures protocol discovery for inbound connections. + + Supersedes the `config.linkerd.io/opaque-ports` annotation. + type: string + default: unknown + additionalPrinterColumns: + - name: Port + type: string + description: The port the server is listening on + jsonPath: .spec.port + - name: Protocol + type: string + description: The protocol of the server + jsonPath: .spec.proxyProtocol + - name: v1beta2 + served: true + storage: false + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + type: object + required: + - port + oneOf: + - required: [podSelector] + - required: [externalWorkloadSelector] + properties: + podSelector: + type: object + description: >- + Selects pods in the same namespace. + + The result of matchLabels and matchExpressions are ANDed. + Selects all if empty. + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + externalWorkloadSelector: + type: object + description: >- + Selects ExternalWorkloads in the same namespace. + + The result of matchLabels and matchExpressions are ANDed. + Selects all if empty. + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + port: + description: >- + A port name or number. Must exist in a pod spec. + x-kubernetes-int-or-string: true + proxyProtocol: + description: >- + Configures protocol discovery for inbound connections. + + Supersedes the `config.linkerd.io/opaque-ports` annotation. + type: string + default: unknown + additionalPrinterColumns: + - name: Port + type: string + description: The port the server is listening on + jsonPath: .spec.port + - name: Protocol + type: string + description: The protocol of the server + jsonPath: .spec.proxyProtocol + - name: v1beta3 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + spec: + type: object + required: + - port + oneOf: + - required: [podSelector] + - required: [externalWorkloadSelector] + properties: + accessPolicy: + type: string + default: deny + description: >- + Default access policy to apply when the traffic doesn't match any of the policy rules. + podSelector: + type: object + description: >- + Selects pods in the same namespace. + + The result of matchLabels and matchExpressions are ANDed. + Selects all if empty. + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + externalWorkloadSelector: + type: object + description: >- + Selects ExternalWorkloads in the same namespace. + + The result of matchLabels and matchExpressions are ANDed. + Selects all if empty. + properties: + matchLabels: + type: object + x-kubernetes-preserve-unknown-fields: true + matchExpressions: + type: array + items: + type: object + required: [key, operator] + properties: + key: + type: string + operator: + type: string + enum: [In, NotIn, Exists, DoesNotExist] + values: + type: array + items: + type: string + port: + description: >- + A port name or number. Must exist in a pod spec. + x-kubernetes-int-or-string: true + proxyProtocol: + description: >- + Configures protocol discovery for inbound connections. + + Supersedes the `config.linkerd.io/opaque-ports` annotation. + type: string + default: unknown + additionalPrinterColumns: + - name: Port + type: string + description: The port the server is listening on + jsonPath: .spec.port + - name: Protocol + type: string + description: The protocol of the server + jsonPath: .spec.proxyProtocol + - name: Access Policy + type: string + description: The default access policy applied when the traffic doesn't match any of the policy rules + jsonPath: .spec.accessPolicy diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/serviceprofile.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/serviceprofile.yaml new file mode 100644 index 0000000000..ad12c96a3a --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/serviceprofile.yaml @@ -0,0 +1,274 @@ +--- +### +### Service Profile CRD +### +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: serviceprofiles.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: linkerd.io + versions: + - name: v1alpha1 + served: true + storage: false + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + description: Spec is the custom resource spec + required: + - routes + properties: + dstOverrides: + type: array + required: + - authority + - weight + items: + type: object + description: WeightedDst is a weighted alternate destination. + properties: + authority: + type: string + weight: + x-kubernetes-int-or-string: true + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + opaquePorts: + type: array + items: + type: string + retryBudget: + type: object + required: + - minRetriesPerSecond + - retryRatio + - ttl + description: RetryBudget describes the maximum number of retries that should be issued to this service. + properties: + minRetriesPerSecond: + format: int32 + type: integer + retryRatio: + type: number + format: float + ttl: + type: string + routes: + type: array + items: + type: object + description: RouteSpec specifies a Route resource. + required: + - condition + - name + properties: + condition: + type: object + description: RequestMatch describes the conditions under which to match a Route. + properties: + pathRegex: + type: string + method: + type: string + all: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + any: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + not: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + isRetryable: + type: boolean + name: + type: string + timeout: + type: string + responseClasses: + type: array + items: + type: object + required: + - condition + description: ResponseClass describes how to classify a response (e.g. success or failures). + properties: + condition: + type: object + description: ResponseMatch describes the conditions under + which to classify a response. + properties: + all: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + any: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + not: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + description: Range describes a range of integers (e.g. status codes). + properties: + max: + format: int32 + type: integer + min: + format: int32 + type: integer + isFailure: + type: boolean + - name: v1alpha2 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + description: Spec is the custom resource spec + properties: + dstOverrides: + type: array + required: + - authority + - weight + items: + type: object + description: WeightedDst is a weighted alternate destination. + properties: + authority: + type: string + weight: + x-kubernetes-int-or-string: true + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + opaquePorts: + type: array + items: + type: string + retryBudget: + type: object + required: + - minRetriesPerSecond + - retryRatio + - ttl + description: RetryBudget describes the maximum number of retries that should be issued to this service. + properties: + minRetriesPerSecond: + format: int32 + type: integer + retryRatio: + type: number + format: float + ttl: + type: string + routes: + type: array + items: + type: object + description: RouteSpec specifies a Route resource. + required: + - condition + - name + properties: + condition: + type: object + description: RequestMatch describes the conditions under which to match a Route. + properties: + pathRegex: + type: string + method: + type: string + all: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + any: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + not: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + isRetryable: + type: boolean + name: + type: string + timeout: + type: string + responseClasses: + type: array + items: + type: object + required: + - condition + description: ResponseClass describes how to classify a response (e.g. success or failures). + properties: + condition: + type: object + description: ResponseMatch describes the conditions under + which to classify a response. + properties: + all: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + any: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + not: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + description: Range describes a range of integers (e.g. status codes). + properties: + max: + format: int32 + type: integer + min: + format: int32 + type: integer + isFailure: + type: boolean + scope: Namespaced + preserveUnknownFields: false + names: + plural: serviceprofiles + singular: serviceprofile + kind: ServiceProfile + shortNames: + - sp diff --git a/charts/buoyant/linkerd-crds/2024.11.3/templates/workload/external-workload.yaml b/charts/buoyant/linkerd-crds/2024.11.3/templates/workload/external-workload.yaml new file mode 100644 index 0000000000..2e6e43ae60 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/templates/workload/external-workload.yaml @@ -0,0 +1,303 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: externalworkloads.workload.linkerd.io + annotations: + {{ include "partials.annotations.created-by" . }} + labels: + helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} + linkerd.io/control-plane-ns: {{.Release.Namespace}} +spec: + group: workload.linkerd.io + names: + categories: + - external + kind: ExternalWorkload + listKind: ExternalWorkloadList + plural: externalworkloads + singular: externalworkload + scope: Namespaced + versions: + - name: v1alpha1 + served: true + storage: false + schema: + openAPIV3Schema: + description: >- + An ExternalWorkload describes a single workload (i.e. a deployable unit) external + to the cluster that should be enrolled in the mesh. + type: object + required: [spec] + properties: + apiVerson: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + meshTls: + description: meshTls describes TLS settings associated with an + external workload. + properties: + identity: + type: string + description: identity of the workload. Corresponds to the + identity used in the workload's certificate. It is used + by peers to perform verification in the mTLS handshake. + minLength: 1 + maxLength: 253 + serverName: + type: string + description: serverName is the name of the workload in DNS + format. It is used by the workload to terminate TLS using + SNI. + minLength: 1 + maxLength: 253 + type: object + required: + - identity + - serverName + ports: + type: array + description: ports describes a list of ports exposed by the + workload + items: + properties: + name: + type: string + description: name must be an IANA_SVC_NAME and unique + within the ports set. Each named port can be referred + to by services. + port: + format: int32 + maximum: 65535 + minimum: 1 + type: integer + protocol: + description: protocol exposed by the port. Must be UDP or + TCP. Defaults to TCP. + type: string + default: "TCP" + type: object + required: + - port + workloadIPs: + type: array + description: workloadIPs contains a list of IP addresses that + can be used to send traffic to the workload. + items: + type: object + properties: + ip: + type: string + # TODO: relax this in the future when ipv6 is supported + # an external workload (like a pod) should only + # support 2 interfaces + maxItems: 1 + type: object + required: + - meshTls + status: + type: object + properties: + conditions: + type: array + items: + type: object + properties: + lastProbeTime: + description: lastProbeTime is the last time the + healthcheck endpoint was probed. + format: date-time + type: string + lastTransitionTime: + description: lastTransitionTime is the last time the + condition transitioned from one status to another. + format: date-time + type: string + status: + description: status of the condition (one of True, False, Unknown) + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of the condition in CamelCase or in + foo.example.com/CamelCase. + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last + transition. Producers of specific condition types may + define expected values and meanings for this field, and + whether the values are considered a guaranteed API. The + value should be a CamelCase string. This field may not + be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + message: + description: message is a human readable message + indicating details about the transition. This may be an + empty string. + maxLength: 32768 + type: string + required: + - status + - type + additionalPrinterColumns: + - jsonPath: .spec.meshTls.identity + name: Identity + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + schema: + openAPIV3Schema: + description: >- + An ExternalWorkload describes a single workload (i.e. a deployable unit) external + to the cluster that should be enrolled in the mesh. + type: object + required: [spec] + properties: + apiVerson: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + meshTLS: + description: meshTLS describes TLS settings associated with an + external workload. + properties: + identity: + type: string + description: identity of the workload. Corresponds to the + identity used in the workload's certificate. It is used + by peers to perform verification in the mTLS handshake. + minLength: 1 + maxLength: 253 + serverName: + type: string + description: serverName is the name of the workload in DNS + format. It is used by the workload to terminate TLS using + SNI. + minLength: 1 + maxLength: 253 + type: object + required: + - identity + - serverName + ports: + type: array + description: ports describes a list of ports exposed by the + workload + items: + properties: + name: + type: string + description: name must be an IANA_SVC_NAME and unique + within the ports set. Each named port can be referred + to by services. + port: + format: int32 + maximum: 65535 + minimum: 1 + type: integer + protocol: + description: protocol exposed by the port. Must be UDP or + TCP. Defaults to TCP. + type: string + default: "TCP" + type: object + required: + - port + workloadIPs: + type: array + description: workloadIPs contains a list of IP addresses that + can be used to send traffic to the workload. This field may + hold a maximum of two entries. If one entry, it can be an + IPv4 or IPv6 address; if two entries it should contain one + IPv4 address and one IPv6 address. + items: + type: object + properties: + ip: + type: string + maxItems: 2 + type: object + required: + - meshTLS + status: + type: object + properties: + conditions: + type: array + items: + type: object + properties: + lastProbeTime: + description: lastProbeTime is the last time the + healthcheck endpoint was probed. + format: date-time + type: string + lastTransitionTime: + description: lastTransitionTime is the last time the + condition transitioned from one status to another. + format: date-time + type: string + status: + description: status of the condition (one of True, False, Unknown) + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: type of the condition in CamelCase or in + foo.example.com/CamelCase. + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + reason: + description: reason contains a programmatic identifier + indicating the reason for the condition's last + transition. Producers of specific condition types may + define expected values and meanings for this field, and + whether the values are considered a guaranteed API. The + value should be a CamelCase string. This field may not + be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + message: + description: message is a human readable message + indicating details about the transition. This may be an + empty string. + maxLength: 32768 + type: string + required: + - status + - type + additionalPrinterColumns: + - jsonPath: .spec.meshTLS.identity + name: Identity + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date diff --git a/charts/buoyant/linkerd-crds/2024.11.3/values.yaml b/charts/buoyant/linkerd-crds/2024.11.3/values.yaml new file mode 100644 index 0000000000..2cc17719e3 --- /dev/null +++ b/charts/buoyant/linkerd-crds/2024.11.3/values.yaml @@ -0,0 +1,3 @@ +enableHttpRoutes: true +enableTlsRoutes: true +enableTcpRoutes: true diff --git a/charts/instana/instana-agent/2.0.2/.helmignore b/charts/instana/instana-agent/2.0.2/.helmignore new file mode 100644 index 0000000000..b4fa6cb0de --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/.helmignore @@ -0,0 +1,23 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +# OWNERS file for helm +OWNERS diff --git a/charts/instana/instana-agent/2.0.2/Chart.yaml b/charts/instana/instana-agent/2.0.2/Chart.yaml new file mode 100644 index 0000000000..722395fe92 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/Chart.yaml @@ -0,0 +1,39 @@ +annotations: + artifacthub.io/links: | + - name: Instana website + url: https://www.ibm.com/products/instana + - name: Instana Helm charts + url: https://github.com/instana/helm-charts + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Instana Agent + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: instana-agent +apiVersion: v2 +appVersion: 1.285.0 +description: Instana Agent for Kubernetes +home: https://www.instana.com/ +icon: file://assets/icons/instana-agent.png +kubeVersion: '>=1.21-0' +maintainers: +- email: felix.marx@ibm.com + name: FelixMarxIBM +- email: henning.treu@ibm.com + name: htreu +- email: konrad.ohms@de.ibm.com + name: Konrad-Ohms +- email: fredrik.gundersen@ibm.com + name: FredrikAtIBM +- email: jefiyamj@ibm.com + name: Jefiya-MJ +- email: milica.cvrkota@ibm.com + name: Milica-Cvrkota-IBM +- email: Nagaraj.Kandoor@ibm.com + name: nagaraj-kandoor +- email: Vineeth.Soman@ibm.com + name: vineethsoman03 +- email: Rashmi.Swamy@ibm.com + name: rashmiswamyibm +name: instana-agent +sources: +- https://github.com/instana/instana-agent-docker +version: 2.0.2 diff --git a/charts/instana/instana-agent/2.0.2/DEPLOYMENT.md b/charts/instana/instana-agent/2.0.2/DEPLOYMENT.md new file mode 100644 index 0000000000..510dea06d8 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/DEPLOYMENT.md @@ -0,0 +1,54 @@ +# Kubernetes Deployment Mode (tech preview) + +Instana has always endeavored to make the experience of using Instana as seamless as possible from auto-instrumentation to one-liner installs. To date for our customers with Kubernetes clusters containing more than 1,000 entities this wasn’t the case. The Kubernetes sensor as a deployment is one of many steps we’re taking to improve the experience of operating Instana in Kubernetes. This is a tech preview however we have a high degree of confidence it will work well in your production workloads. The fundamental change moves the Kubernetes sensor from the DaemonSet responsible for monitoring your hosts and processes into its own dedicated Deployment where it does not contend for resources with other sensors. An overview of this deployment is below: + +![kubernetes.deployment.enabled=true](kubernetes.deployment.enabled.png) + +This change provides a few primary benefits including: + +* Lower load on the Kubernetes api-server as it eliminates per node pod monitoring. +* Lower load on the Kubernetes api-server as it reduces the endpoint watch to 2 leader elector side cars. +* Lower memory and CPU requests in the DaemonSet as it is no longer responsible for monitoring Kubernetes. +* Elimination of the leader elector sidecar in the DaemonSet as it is only required for the Kubernetes sensor. +* Better performance of the Kubernetes sensor as it is isolated from other sensors and does not contend for CPU and memory. +* Better scaling behaviour as you can adjust the memory and CPU requirements to monitor your clusters without overprovisioning utilisation cluster wide. + +The primary drawback of this model in the tech preview include: + +* Reduced control and observability of the Kubernetes specific Agents in the Agent dashboard. +* Some unnecessary features are still enabled in the Kubernetes sensor (e.g. trace sinks, and host monitoring). + +Some limitations remain unchanged from the previous sensor: + +* Clusters with a high number of entities (e.g. pods, deployments, etc) are likely to have non-deterministic behaviour due to limitations we impose on message sizes. This is unlikely to be experienced in clusters with fewer than 500 hosts. +* The ServiceAccount is shared between both the DaemonSet and Deployment meaning no change in the security posture. We plan to add an additional service account to limit access to the api-server to only the Kubernetes sensor Deployment. + +## Installation + +For clusters with minimal controls you can install the tech preview with the following Helm install command: + +``` +helm template instana-agent \ + --repo https://agents.instana.io/helm \ + --namespace instana-agent \ + --create-namespace \ + --set agent.key=${AGENT_KEY} \ + --set agent.endpointHost=${BACKEND_URL} \ + --set agent.endpointPort=443 \ + --set cluster.name=${CLUSTER_NAME} \ + --set zone.name=${ZONE_NAME} \ + --set kubernetes.deployment.enabled=true \ + instana-agent +``` + +If your cluster employs Pod Security Policies you will need the following additional flag: + +``` +--set podSecurityPolicy.enable=true +``` + +If you are deploying into an OpenShift 4.x cluster you will need the following additional flag: + +``` +--set openshift=true +``` diff --git a/charts/instana/instana-agent/2.0.2/README.md b/charts/instana/instana-agent/2.0.2/README.md new file mode 100644 index 0000000000..eb88617b03 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/README.md @@ -0,0 +1,701 @@ +# Instana + +Instana is an [APM solution](https://www.instana.com/) built for microservices that enables IT Ops to build applications faster and deliver higher quality services by automating monitoring, tracing and root cause analysis. +This solution is optimized for [Kubernetes](https://www.instana.com/automatic-kubernetes-monitoring/). + +This chart adds the Instana Agent to all schedulable nodes in your cluster via a privileged `DaemonSet` and accompanying resources like `ConfigurationMap`s, `Secret`s and RBAC settings. + +## Prerequisites + +* Kubernetes 1.21+ OR OpenShift 4.8+ +* Helm 3 + +## Installation + +To configure the installation you can either specify the options on the command line using the **--set** switch, or you can edit **values.yaml**. + +First, create a namespace for the instana-agent + +```bash +$ kubectl create namespace instana-agent +``` + +**OpenShift:** When targetting an OpenShift 4.x cluster, ensure proper permission before installing the helm chart, otherwise the agent pods will not be scheduled correctly. + +```bash +$ oc adm policy add-scc-to-user privileged -z instana-agent +``` + +To install the chart with the release name `instana-agent` and set the values on the command line run: + +```bash +$ helm install instana-agent \ +--namespace instana-agent \ +--repo https://agents.instana.io/helm \ +--set agent.key=INSTANA_AGENT_KEY \ +--set agent.endpointHost=HOST \ +--set zone.name=ZONE_NAME \ +--set cluster.name="CLUSTER_NAME" \ +instana-agent +``` + +## Upgrade + +The helm chart deploys a Kubernetes Operator internally that reconciles the agent resources based on an agent CustomResource (CR) that is created based on the helm values. As the Operator pattern requires a CustomResourceDefinition (CRD) to be present in the cluster before defining any CRs, the CRD definition is included in the helm chart. On initial installations the chart deploys the CRD before submitting the rest of the artifacts. + +Helm has known limitations around handling the CRD lifecycle as outlined in their [documentation](https://helm.sh/docs/chart_best_practices/custom_resource_definitions/). + +This leads to the problem, that CRD updates are only submitted to the Cluster on **initial installation**, but **NOT applied automatically during upgrades**. + +It is also worth noting, that CRDs must be removed manually if the chart should be removed completly, see more details in the [uninstall](#uninstallation) section. + +To ensure a proper update, apply the CRD updates before running the upgrade: + +``` +helm pull --repo https://agents.instana.io/helm --untar instana-agent; kubectl apply -f instana-agent/crds; helm upgrade instana-agent instana-agent \ + --namespace instana-agent \ + --repo https://agents.instana.io/helm \ + --reuse-values +``` + +This is especially important when migrating from helm charts v1 to v2, as the upgrade will fail otherwise as the CR artifact cannot be created. + +### Required Settings + +#### Configuring the Instana Backend + +In order to report the data it collects to the Instana backend for analysis, the Instana agent must know which backend to report to, and which credentials to use to authenticate, known as "agent key". + +As described by the [Install Using the Helm Chart](https://www.instana.com/docs/setup_and_manage/host_agent/on/kubernetes#install-using-the-helm-chart) documentation, you will find the right values for the following fields inside Instana itself: + +* `agent.endpointHost` +* `agent.endpointPort` +* `agent.key` + +_Note:_ You can find the options mentioned in the [configuration section below](#configuration-reference) + +If your agents report into a self-managed Instana unit (also known as "on-prem"), you will also need to configure a "download key", which allows the agent to fetch its components from the Instana repository. +The download key is set via the following value: + +* `agent.downloadKey` + +#### Zone and Cluster + +Instana needs to know how to name your Kubernetes cluster and, optionally, how to group your Instana agents in [Custom zones](https://www.instana.com/docs/setup_and_manage/host_agent/configuration/#custom-zones) using the following fields: + +* `zone.name` +* `cluster.name` + +Either `zone.name` or `cluster.name` are required. +If you omit `cluster.name`, the value of `zone.name` will be used as cluster name as well. +If you omit `zone.name`, the host zone will be automatically determined by the availability zone information provided by the [supported Cloud providers](https://www.instana.com/docs/setup_and_manage/cloud_service_agents). + +## Uninstallation + +To uninstall/delete the `instana-agent` release: + +```bash +helm uninstall instana-agent -n instana-agent && kubectl patch agent instana-agent -p '{"metadata":{"finalizers":null}}' --type=merge && +kubectl delete crd/agents.instana.io +``` + +## Configuration Reference + +The following table lists the configurable parameters of the Instana chart and their default values. + +| Parameter | Description | Default | +| --------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------- | +| `agent.configuration_yaml` | Custom content for the agent configuration.yaml file | `nil` See [below](#agent-configuration) for more details | +| `agent.endpointHost` | Instana Agent backend endpoint host | `ingress-red-saas.instana.io` (US and ROW). If in Europe, please override with `ingress-blue-saas.instana.io` | +| `agent.endpointPort` | Instana Agent backend endpoint port | `443` | +| `agent.key` | Your Instana Agent key | `nil` You must provide your own key unless `agent.keysSecret` is specified | +| `agent.downloadKey` | Your Instana Download key | `nil` Usually not required | +| `agent.keysSecret` | As an alternative to specifying `agent.key` and, optionally, `agent.downloadKey`, you can instead specify the name of the secret in the namespace in which you install the Instana agent that carries the agent key and download key | `nil` Usually not required, see [Bring your own Keys secret](#bring-your-own-keys-secret) for more details | +| `agent.additionalBackends` | List of additional backends to report to; it must specify the `endpointHost` and `key` fields, and optionally `endpointPort` | `[]` Usually not required; see [Configuring Additional Backends](#configuring-additional-backends) for more info and examples | +| `agent.tls.secretName` | The name of the secret of type `kubernetes.io/tls` which contains the TLS relevant data. If the name is provided, `agent.tls.certificate` and `agent.tls.key` will be ignored. | `nil` | +| `agent.tls.certificate` | The certificate data encoded as base64. Which will be used to create a new secret of type `kubernetes.io/tls`. | `nil` | +| `agent.tls.key` | The private key data encoded as base64. Which will be used to create a new secret of type `kubernetes.io/tls`. | `nil` | +| `agent.image.name` | The image name to pull | `instana/agent` | +| `agent.image.digest` | The image digest to pull; if specified, it causes `agent.image.tag` to be ignored | `nil` | +| `agent.image.tag` | The image tag to pull; this property is ignored if `agent.image.digest` is specified | `latest` | +| `agent.image.pullPolicy` | Image pull policy | `Always` | +| `agent.image.pullSecrets` | Image pull secrets; if not specified (default) _and_ `agent.image.name` starts with `containers.instana.io`, it will be automatically set to `[{ "name": "containers-instana-io" }]` to match the default secret created in this case. | `nil` | +| `agent.listenAddress` | List of addresses to listen on, or "*" for all interfaces | `nil` | +| `agent.mode` | Agent mode. Supported values are `APM`, `INFRASTRUCTURE`, `AWS` | `APM` | +| `agent.instanaMvnRepoUrl` | Override for the Maven repository URL when the Agent needs to connect to a locally provided Maven repository 'proxy' | `nil` Usually not required | +| `agent.instanaMvnRepoFeaturesPath` | Override for the Maven repository features path the Agent needs to connect to a locally provided Maven repository 'proxy' | `nil` Usually not required | +| `agent.instanaMvnRepoSharedPath` | Override for the Maven repository shared path when the Agent needs to connect to a locally provided Maven repository 'proxy' | `nil` Usually not required | +| `agent.updateStrategy.type` | [DaemonSet update strategy type](https://kubernetes.io/docs/tasks/manage-daemon/update-daemon-set/); valid values are `OnDelete` and `RollingUpdate` | `RollingUpdate` | +| `agent.updateStrategy.rollingUpdate.maxUnavailable` | How many agent pods can be updated at once; this value is ignored if `agent.updateStrategy.type` is different than `RollingUpdate` | `1` | +| `agent.minReadySeconds` | The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available | `0` | +| `agent.pod.annotations` | Additional annotations to apply to the pod | `{}` | +| `agent.pod.labels` | Additional labels to apply to the Agent pod | `{}` | +| `agent.pod.priorityClassName` | Name of an _existing_ PriorityClass that should be set on the agent pods | `nil` | +| `agent.proxyHost` | Hostname/address of a proxy | `nil` | +| `agent.proxyPort` | Port of a proxy | `nil` | +| `agent.proxyProtocol` | Proxy protocol. Supported proxy types are `http` (for both HTTP and HTTPS proxies), `socks4`, `socks5`. | `nil` | +| `agent.proxyUser` | Username of the proxy auth | `nil` | +| `agent.proxyPassword` | Password of the proxy auth | `nil` | +| `agent.proxyUseDNS` | Boolean if proxy also does DNS | `nil` | +| `agent.pod.limits.cpu` | Container cpu limits in cpu cores | `1.5` | +| `agent.pod.limits.memory` | Container memory limits in MiB | `768Mi` | +| `agent.pod.requests.cpu` | Container cpu requests in cpu cores | `0.5` | +| `agent.pod.requests.memory` | Container memory requests in MiB | `768Mi` | +| `agent.pod.tolerations` | Tolerations for pod assignment | `[]` | +| `agent.pod.affinity` | Affinity for pod assignment | `{}` | +| `agent.serviceMesh.enabled` | Activate Instana Agent JVM monitoring service mesh support for Istio or OpenShift ServiceMesh | `true` | +| `agent.env` | Additional environment variables for the agent | `{}` | +| `agent.redactKubernetesSecrets` | Enable additional secrets redaction for selected Kubernetes resources | `nil` See [Kubernetes secrets](https://docs.instana.io/setup_and_manage/host_agent/on/kubernetes/#secrets) for more details. | +| `cluster.name` | Display name of the monitored cluster | Value of `zone.name` | +| `k8s_sensor.deployment.enabled` | Isolate k8sensor with a deployment | +| `k8s_sensor.deployment.minReadySeconds` | The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available | `0` | `true` | +| `k8s_sensor.image.name` | The k8sensor image name to pull | `gcr.io/instana/k8sensor` | +| `k8s_sensor.image.digest` | The image digest to pull; if specified, it causes `k8s_sensor.image.tag` to be ignored | `nil` | +| `k8s_sensor.image.tag` | The image tag to pull; this property is ignored if `k8s_sensor.image.digest` is specified | `latest` | +| `k8s_sensor.deployment.pod.limits.cpu` | CPU request for the `k8sensor` pods | `4` | +| `k8s_sensor.deployment.pod.limits.memory` | Memory request limits for the `k8sensor` pods | `6144Mi` | +| `k8s_sensor.deployment.pod.requests.cpu` | CPU limit for the `k8sensor` pods | `1.5` | +| `k8s_sensor.deployment.pod.requests.memory` | Memory limit for the `k8sensor` pods | `1024Mi` | +| `podSecurityPolicy.enable` | Whether a PodSecurityPolicy should be authorized for the Instana Agent pods. Requires `rbac.create` to be `true` as well and it is available until Kubernetes version v1.25. | `false` See [PodSecurityPolicy](https://docs.instana.io/setup_and_manage/host_agent/on/kubernetes/#podsecuritypolicy) for more details. | +| `podSecurityPolicy.name` | Name of an _existing_ PodSecurityPolicy to authorize for the Instana Agent pods. If not provided and `podSecurityPolicy.enable` is `true`, a PodSecurityPolicy will be created for you. | `nil` | +| `rbac.create` | Whether RBAC resources should be created | `true` | +| `opentelemetry.grpc.enabled` | Whether to configure the agent to accept telemetry from OpenTelemetry applications via gRPC. This option also implies `service.create=true`, and requires Kubernetes 1.21+, as it relies on `internalTrafficPolicy`. | `true` | +| `opentelemetry.http.enabled` | Whether to configure the agent to accept telemetry from OpenTelemetry applications via HTTP. This option also implies `service.create=true`, and requires Kubernetes 1.21+, as it relies on `internalTrafficPolicy`. | `true` | +| `prometheus.remoteWrite.enabled` | Whether to configure the agent to accept metrics over its implementation of the `remote_write` Prometheus endpoint. This option also implies `service.create=true`, and requires Kubernetes 1.21+, as it relies on `internalTrafficPolicy`. | `false` | +| `service.create` | Whether to create a service that exposes the agents' Prometheus, OpenTelemetry and other APIs inside the cluster. Requires Kubernetes 1.21+, as it relies on `internalTrafficPolicy`. The `ServiceInternalTrafficPolicy` feature gate needs to be enabled (default: enabled). | `true` | +| `serviceAccount.create` | Whether a ServiceAccount should be created | `true` | +| `serviceAccount.name` | Name of the ServiceAccount to use | `instana-agent` | +| `serviceAccount.annotations` | Annotations to add to the service account | `{}` | +| `zone.name` | Zone that detected technologies will be assigned to | `nil` You must provide either `zone.name` or `cluster.name`, see [above](#installation) for details | +| `zones` | Multi-zone daemonset configuration. | `nil` see [below](#multiple-zones) for details | +| `k8s_sensor.podDisruptionBudget.enabled` | Whether to create DisruptionBudget for k8sensor to limit the number of concurrent disruptions | `false` | +| `k8s_sensor.deployment.pod.affinity` | `k8sensor` deployment affinity format | `podAntiAffinity` defined in `values.yaml` | + +### Agent Modes + +Agent can have either `APM` or `INFRASTRUCTURE`. +Default is APM and if you want to override that, ensure you set value: + +* `agent.mode` + +For more information on agent modes, refer to the [Host Agent Modes](https://www.instana.com/docs/setup_and_manage/host_agent#host-agent-modes) documentation. + +### Agent Configuration + +Besides the settings listed above, there are many more settings that can be applied to the agent via the so-called "Agent Configuration File", often also referred to as `configuration.yaml` file. +An overview of the settings that can be applied is provided in the [Agent Configuration File](https://www.instana.com/docs/setup_and_manage/host_agent/configuration#agent-configuration-file) documentation. +To configure the agent, you can either: + +* edit the [config map](templates/agent-configmap.yaml), or +* provide the configuration via the `agent.configuration_yaml` parameter in [values.yaml](values.yaml) + +This configuration will be used for all Instana Agents on all nodes. Visit the [agent configuration documentation](https://docs.instana.io/setup_and_manage/host_agent/#agent-configuration-file) for more details on configuration options. + +_Note:_ This Helm Chart does not support configuring [Multiple Configuration Files](https://www.instana.com/docs/setup_and_manage/host_agent/configuration#multiple-configuration-files). + +### Agent Pod Sizing + +The `agent.pod.requests.cpu`, `agent.pod.requests.memory`, `agent.pod.limits.cpu` and `agent.pod.limits.memory` settings allow you to change the sizing of the `instana-agent` pods. +If you are using the [Kubernetes Sensor Deployment](#kubernetes-sensor-deployment) functionality, you may be able to reduce the default amount of resources, and especially memory, allocated to the Instana agents that monitor your applications. +Actual sizing data depends very much on how many pods, containers and applications are monitored, and how much traces they generate, so we cannot really provide a rule of thumb for the sizing. + +### Bring your own Keys secret + +In case you have automation that creates secrets for you, it may not be desirable for this Helm chart to create a secret containing the `agent.key` and `agent.downloadKey`. +In this case, you can instead specify the name of an alread-existing secret in the namespace in which you install the Instana agent that carries the agent key and download key. + +The secret you specify The secret you specify _must_ have a field called `key`, which would contain the value you would otherwise set to `agent.key`, and _may_ contain a field called `downloadKey`, which would contain the value you would otherwise set to `agent.downloadKey`. + +### Configuring Additional Configuration Files + +[Multiple configuration files](https://www.instana.com/docs/setup_and_manage/host_agent/configuration#multiple-configuration-files) is a capability of the Instana agent that allows for modularity in its configurations files. + +The experimental `agent.configuration.autoMountConfigEntries`, which uses functionality available in Helm 3.1+ to automatically look up the entries of the default `instana-agent` ConfigMap, and mount as agent configuration files in the `instana-agent` container under the `/opt/instana/agent/etc/instana` directory all ConfigMap entries with keys that match the `configuration-*.yaml` scheme. + +**IMPORTANT:** Needs Helm 3.1+ as it is built on the `lookup` function +**IMPORTANT:** Editing the ConfigMap adding keys requires a `helm upgrade` to take effect + +### Configuring Additional Backends + +You may want to have your Instana agents report to multiple backends. +The first backend must be configured as shown in the [Configuring the Instana Backend](#configuring-the-instana-backend); every backend after the first, is configured in the `agent.additionalBackends` list in the [values.yaml](values.yaml) as follows: + +```yaml +agent: + additionalBackends: + # Second backend + - endpointHost: my-instana.instana.io # endpoint host; e.g., my-instana.instana.io + endpointPort: 443 # default is 443, so this line could be omitted + key: ABCDEFG # agent key for this backend + # Third backend + - endpointHost: another-instana.instana.io # endpoint host; e.g., my-instana.instana.io + endpointPort: 1444 # default is 443, so this line could be omitted + key: LMNOPQR # agent key for this backend +``` + +The snippet above configures the agent to report to two additional backends. +The same effect as the above can be accomplished via the command line via: + +```sh +$ helm install -n instana-agent instana-agent ... \ + --repo https://agents.instana.io/helm \ + --set 'agent.additionalBackends[0].endpointHost=my-instana.instana.io' \ + --set 'agent.additionalBackends[0].endpointPort=443' \ + --set 'agent.additionalBackends[0].key=ABCDEFG' \ + --set 'agent.additionalBackends[1].endpointHost=another-instana.instana.io' \ + --set 'agent.additionalBackends[1].endpointPort=1444' \ + --set 'agent.additionalBackends[1].key=LMNOPQR' \ + instana-agent +``` + +_Note:_ There is no hard limitation on the number of backends an Instana agent can report to, although each comes at the cost of a slight increase in CPU and memory consumption. + +### Configuring a Proxy between the Instana agents and the Instana backend + +If your infrastructure uses a proxy, you should ensure that you set values for: + +* `agent.proxyHost` +* `agent.pod.proxyPort` +* `agent.pod.proxyProtocol` +* `agent.pod.proxyUser` +* `agent.pod.proxyPassword` +* `agent.pod.proxyUseDNS` + +#### Same Proxy for Repository and the Instana backend + +If the same proxy is utilized for both backend and repository, configure only the 'Agent' proxy settings using the following parameter: + ``` + --set agent.proxyHost='' + ``` + +#### Separate Proxies for Repository and the Instana backend + +In scenarios where distinct proxy settings are employed for the backend and repository, both proxies must be configured separately. The key is to ensure that `INSTANA_REPOSITORY_PROXY_ENABLED=true` is set. + +To use this variant, execute helm install with the following additional parameters: + +``` +--set agent.proxyHost='Hostname/address of a proxy' +--set agent.env.INSTANA_REPOSITORY_PROXY_ENABLED='true' +--set agent.env.INSTANA_REPOSITORY_PROXY_HOST='Hostname/address of a proxy' +``` +Make sure to replace 'Hostname/address of a proxy' with the actual hostname or address of your proxy. + +### Configuring which Networks the Instana Agent should listen on + +If your infrastructure has multiple networks defined, you might need to allow the agent to listen on all addresses (typically with value set to `*`): + +* `agent.listenAddress` + +### Setup TLS Encryption for Agent Endpoint + +TLS encryption can be added via two variants. +Either an existing secret can be used or a certificate and a private key can be used during the installation. + +#### Using existing secret + +An existing secret of type `kubernetes.io/tls` can be used. +Only the `secretName` must be provided during the installation with `--set 'agent.tls.secretName='`. +The files from the provided secret are then mounted into the agent. + +#### Provide certificate and private key + +On the other side, a certificate and a private key can be added during the installation. +The certificate and private key must be base64 encoded. + +To use this variant, execute `helm install` with the following additional parameters: + +``` +--set 'agent.tls.certificate=' +--set 'agent.tls.key=' +``` + +If `agent.tls.secretName` is set, then `agent.tls.certificate` and `agent.tls.key` are ignored. + +### Development and debugging options + +These options will be rarely used outside of development or debugging of the agent. + +| Parameter | Description | Default | +| ----------------------- | ------------------------------------------------ | ------- | +| `agent.host.repository` | Host path to mount as the agent maven repository | `nil` | + +### Kubernetes Sensor Deployment + + _Note: leader-elector and kubernetes sensor are fully deprecated and can no longer be chosen. Instead, the k8s_sensor will be used._ + +The Helm chart will schedule additional Instana agents running _only_ the Kubernetes sensor that runs in a dedicated `k8sensor` Deployment inside the `instana-agent` namespace. +The pods containing agents that run only the Kubernetes sensor are called `k8sensor` pods. +When `k8s_sensor.deployment.enabled=true`, the `instana-agent` pods running inside the daemonset do _not_ contain the `leader-elector` container, which is instead scheduled inside the `k8sensor` pods. + +The `instana-agent` and `k8sensor` pods share the same configurations in terms of backend-related configurations (including [additional backends](#configuring-additional-backends)). + +The `k8s_sensor.deployment.pod.requests.cpu`, `k8s_sensor.deployment.pod.requests.memory`, `k8s_sensor.deployment.pod.limits.cpu` and `k8s_sensor.deployment.pod.limits.memory` settings, allow you to change the sizing of the `k8sensor` pods. + +### Multiple Zones + +You can list zones to use affinities and tolerations as the basis to associate a specific daemonset per tainted node pool. Each zone will have the following data: + +* `name` (required) - zone name. +* `mode` (optional) - instana agent mode (e.g. APM, INFRASTRUCTURE, etc). +* `affinity` (optional) - standard kubernetes pod affinity list for the daemonset. +* `tolerations` (optional) - standard kubernetes pod toleration list for the daemonset. + +The following is an example that will create 2 zones an api-server and a worker zone: + +```yaml +zones: + - name: workers + mode: APM + - name: api-server + mode: INFRASTRUCTURE + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: node-role.kubernetes.io/control-plane + operator: Exists + tolerations: + - key: "node-role.kubernetes.io/master" + operator: "Exists" + effect: "NoSchedule" +``` + +## Changelog + +### 2.0.1 + +* Fix rendering of the `spec.agent.env`, `spec.configuration_yaml`, `spec.agent.image.pullSecrets` + +### 2.0.0 + +* Deploy the instana-agent operator instead of managing agent artifacts directly +* Always use the k8sensor, the deprecated kubernetes sensor is no longer supported (this is an internal change, Kubernetes clusters will still report into the Instana backend) +* BREAKING CHANGE: Due to limitations of helm to manage Custom Resource Definition (CRD) updates, the upgrade requires to apply the CRD from the helm chart crds folder manually. Find more details in the [upgrade](#upgrade) section. + +### 1.2.74 + +* Enable OTLP by default + +### 1.2.73 + +* Fix label for `io.instana/zone` to reflect the real agent mode +* Change the charts flag from ENABLE_AGENT_SOCKET to serviceMesh.enabled +* Add type: DirectoryOrCreate to DaemonSet definitions to ensure required directories exist + +### 1.2.72 + +* Add minReadySeconds field to agent daemonset yaml + +### 1.2.71 + +* Fix usage of digest for pulling images + +### 1.2.70 + +* Allow the configuration of `minReadySeconds` for the agent daemonset and deployment + +### 1.2.69 + +* Add possibility to set annotations for the serviceAccount. + +### 1.2.68 + +* Add leader elector configuration back to allow for proper deprecation + +### 1.2.67 + +* Fix variable name in the K8s deployment + +### 1.2.66 + +* Allign the default Memory requests to 768Mi for the Agent container. + +### 1.2.65 + +* Ensure we have appropriate SCC when running with new K8s sensor. + +### 1.2.64 + +* Remove RBAC not required by agent when kubernetes-sensor is disabed. +* Add settings override for k8s-sensor affinity +* Add optional pod disruption budget for k8s-sensor + +### 1.2.63 + +* Add RBAC required to allow access to /metrics end-points. + +### 1.2.62 + +* Include k8s-sensor resources in the default static YAML definitions + +### 1.2.61 + +* Increase timeout and initialDelay for the Agent container +* Add OTLP ports to headless service + +### 1.2.60 + +* Enable the k8s_sensor by default + +### 1.2.59 + +* Introduce unique selectorLabels and commonLabels for k8s-sensor deployment + +### 1.2.58 + +* Default to `internalTrafficPolicy` instead of `topologyKeys` for rendering of static YAMLs + +### 1.2.57 + +* Fix vulnerability in the leader-elector image + +### 1.2.49 + +* Add zone name to label `io.instana/zone` in daemonset + +### 1.2.48 + +* Set env var INSTANA_KUBERNETES_REDACT_SECRETS true if agent.redactKubernetesSecrets is enabled. +* Use feature PSP flag in k8sensor ClusterRole only when podsecuritypolicy.enable is true. + +### 1.2.47 + +* Roll back the changes from version 1.2.46 to be compatible with the Agent Operator installation + +### 1.2.46 + +* Use K8sensor by default. +* kubernetes.deployment.enabled setting overrides k8s_sensor.deployment.enabled setting. +* Use feature PSP flag in k8sensor ClusterRole only when podsecuritypolicy.enable is true. +* Throw failure if customer specifies proxy with k8sensor. +* Set env var INSTANA_KUBERNETES_REDACT_SECRETS true if agent.redactKubernetesSecrets is enabled. + +### 1.2.45 + +* Use agent key secret in k8sensor deployment. + +### 1.2.44 + +* Add support for enabling the hot-reload of `configuration.yaml` when the default `instana-agent` ConfigMap changes +* Enablement is done via the flag `--set agent.configuration.hotreloadEnabled=true` + +### 1.2.43 + +* Bump leader-elector image to v0.5.16 (Update dependencies) + +### 1.2.42 + +* Add support for creating multiple zones within the same cluster using affinity and tolerations. + +### 1.2.41 + +* Add additional permissions (HPA, ResourceQuotas, etc) to k8sensor clusterrole. + +### 1.2.40 + +* Mount all system mounts mountPropagation: HostToContainer. + +### 1.2.39 + +* Add NO_PROXY to k8sensor deployment to prevent api-server requests from being routed to the proxy. + +### 1.2.38 + +* Fix issue related to EKS version format when enabling OTel service. + +### 1.2.37 + +* Fix issue where cluster_zone is used as cluster_name when `k8s_sensor.deployment.enabled=true`. +* Set `HTTPS_PROXY` in k8s deployment when proxy information is set. + +### 1.2.36 + +* Remove Service `topologyKeys`, which was removed in Kubernetes v1.22. Replaced by `internalTrafficPolicy` which is available with Kubernetes v1.21+. + +### 1.2.35 + +* Fix invalid backend port for new Kubernetes sensor (k8sensor) + +### 1.2.34 + +* Add support for new Kubernetes sensor (k8sensor) + * New Kubernetes sensor can be used via the flag `--set k8s_sensor.deployment.enabled=true` + +### 1.2.33 + +* Bump leader-elector image to v0.5.15 (Update dependencies) + +### 1.2.32 + +* Add support for containerd montoring on TKGI + +### 1.2.31 + +* Bump leader-elector image to v0.5.14 (Update dependencies) + +### 1.2.30 + +* Pull agent image from IBM Cloud Container Registry (icr.io/instana/agent). No code changes have been made. +* Bump leader-elector image to v0.5.13 and pull from IBM Cloud Container Registry (icr.io/instana/leader-elector). No code changes have been made. + +### 1.2.29 + +* Add an additional port to the Instana Agent `Service` definition, for the OpenTelemetry registered IANA port 4317. + +### 1.2.28 + +* Fix deployment when `cluster.name` is not specified. Should be allowed according to docs but previously broke the Pod + when starting up. + +### 1.2.27 + +* Update leader elector image to `0.5.10` to tone down logging and make it configurable + +### 1.2.26 + +* Add TLS support. An existing secret can be used of type `kubernetes.io/tls`. Or provide a certificate and a private key, which creates a new secret. +* Update leader elector image version to 0.5.9 to support PPCle + +### 1.2.25 + +* Add `agent.pod.labels` to add custom labels to the Instana Agent pods + +### 1.2.24 + +* Bump leader-elector image to v0.5.8 which includes a health-check endpoint. Update the `livenessProbe` + correspondingly. + +### 1.2.23 + +* Bump leader-elector image to v0.5.7 to fix a potential Golang bug in the elector + +### 1.2.22 + +* Fix templating scope when defining multiple backends + +### 1.2.21 + +* Internal updates + +### 1.2.20 + +* upgrade leader-elector image to v0.5.6 to enable usage on s390x and arm64 + +### 1.2.18 / 1.2.19 + +* Internal change on generated DaemonSet YAML from the Helm charts + +### 1.2.17 + +* Update Pod Security Policies as the `readOnly: true` appears not to be working for the mount points and + actually causes the Agent deployment to fail when these policies are enforced in the cluster. + +### 1.2.16 + +* Add configuration option for `INSTANA_MVN_REPOSITORY_URL` setting on the Agent container. + +### 1.2.15 + +* Internal pipeline changes. No significant changes to the Helm charts + +### v1.2.14 + +* Update Agent container mounts. Make some read-only as we don't need all mounts with read-write permissions. + Additionally add the mount for `/var/data` which is needed in certain environments for the Agent to function + properly. + +### v1.2.13 + +* Update memory settings specifically for the Kubernetes sensor (Technical Preview) + +### v1.2.11 + +* Simplify setup for using OpenTelemetry and the Prometheus `remote_write` endpoint using the `opentelemetry.enabled` and `prometheus.remoteWrite.enabled` settings, respectively. + +### v1.2.9 + +* **Technical Preview:** Introduce a new mode of running to the Kubernetes sensor using a dedicated deployment. + See the [Kubernetes Sensor Deployment](#kubernetes-sensor-deployment) section for more information. + +### v1.2.7 + +* Fix: Make service opt-in, as it uses functionality (`topologyKeys`) that is available only in K8S 1.17+. + +### v1.2.6 + +* Fix bug that might cause some OpenShift-specific resources to be created in other flavours of Kubernetes. + +### v1.2.5 + +* Introduce the `instana-agent:instana-agent` Kubernetes service that allows you to talk to the Instana agent on the same node. + +### v1.2.3 + +* Bug fix: Extend the built-in Pod Security Policy to cover the Docker socket mount for Tanzu Kubernetes Grid systems. + +### v1.2.1 + +* Support OpenShift 4.x: just add --set openshift=true to the usual settings, and off you go :-) +* Restructure documentation for consistency and readability +* Deprecation: Helm 2 is no longer supported; the minimum Helm API version is now v2, which will make Helm 2 refuse to process the chart. + +### v1.1.10 + +* Some linting of the whitespaces in the generated YAML + +### v1.1.9 + +* Update the README to replace all references of `stable/instana-agent` with specifically setting the repo flag to `https://agents.instana.io/helm`. +* Add support for TKGI and PKS systems, providing a workaround for the [unexpected Docker socket location](https://github.com/cloudfoundry-incubator/kubo-release/issues/329). + +### v1.1.7 + +* Store the cluster name in a new `cluster-name` entry of the `instana-agent` ConfigMap rather than directly as the value of the `INSTANA_KUBERNETES_CLUSTER_NAME`, so that you can edit the cluster name in the ConfigMap in deployments like VMware Tanzu Kubernetes Grid in which, when installing the Instana agent over the [Instana tile](https://www.instana.com/docs/setup_and_manage/host_agent/on/vmware_tanzu), you do not have directly control to the configuration of the cluster name. +If you edit the ConfigMap, you will need to delete the `instana-agent` pods for its new value to take effect. + +### v1.1.6 + +* Allow to use user-specified memony measurement units in `agent.pod.requests.memory` and `agent.pod.limits.memory`. + If the value set is numerical, the Chart will assume it to be expressed in `Mi` for backwards compatibility. +* Exposed `agent.updateStrategy.type` and `agent.updateStrategy.rollingUpdate.maxUnavailable` settings. + +### v1.1.5 + +Restore compatibility with Helm 2 that was broken in v1.1.4 by the usage of the `lookup` function, a function actually introduced only with Helm 3.1. +Coincidentally, this has been an _excellent_ opportunity to introduce `helm lint` to our validation pipeline and end-to-end tests with Helm 2 ;-) + +### v1.1.4 + +* Bring-your-own secret for agent keys: using the new `agent.keysSecret` setting, you can specify the name of the secret that contains the agent key and, optionally, the download key; refer to [Bring your own Keys secret](#bring-your-own-keys-secret) for more details. +* Add support for affinities for the instana agent pod via the `agent.pod.affinity` setting. +* Put some love into the ArtifactHub.io metadata; likely to add some more improvements related to this over time. + +### v1.1.3 + +* No new features, just ironing some wrinkles out of our release automation. + +### v1.1.2 + +* Improvement: Seamless support for Instana static agent images: When using an `agent.image.name` starting with `containers.instana.io`, automatically create a secret called `containers-instana-io` containing the `.dockerconfigjson` for `containers.instana.io`, using `_` as username and `agent.downloadKey` or, if missing, `agent.key` as password. If you want to control the creation of the image pull secret, or disable it, you can use `agent.image.pullSecrets`, passing to it the YAML to use for the `imagePullSecrets` field of the Daemonset spec, including an empty array `[]` to mount no pull secrets, no matter what. + +### v1.1.1 + +* Fix: Recreate the `instana-agent` pods when there is a change in one of the following configuration, which are mapped to the chart-managed ConfigMap: + +* `agent.configuration_yaml` +* `agent.additional_backends` + +The pod recreation is achieved by annotating the `instana-agent` Pod with a new `instana-configuration-hash` annotation that has, as value, the SHA-1 hash of the configurations used to populate the ConfigMap. +This way, when the configuration changes, the respective change in the `instana-configuration-hash` annotation will cause the agent pods to be recreated. +This technique has been described at [1] (or, at least, that is were we learned about it) and it is pretty cool :-) + +### v1.1.0 + +* Improvement: The `instana-agent` Helm chart has a new home at `https://agents.instana.io/helm` and `https://github.com/instana/helm-charts/instana-agent`! +This release is functionally equivalent to `1.0.34`, but we bumped the major to denote the new location ;-) + +## References + +[1] ["Using Kubernetes Helm to push ConfigMap changes to your Deployments", by Sander Knape; Mar 7, 2019](https://sanderknape.com/2019/03/kubernetes-helm-configmaps-changes-deployments/) diff --git a/charts/instana/instana-agent/2.0.2/app-readme.md b/charts/instana/instana-agent/2.0.2/app-readme.md new file mode 100644 index 0000000000..0e26c86231 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/app-readme.md @@ -0,0 +1,5 @@ +# Instana + +Instana is an [APM solution(https://www.instana.com/) built for microservices that enables IT Ops to build applications faster and deliver higher quality services by automating monitoring tracing and root cause analysis. This solution is optimized for [Rancher](https://www.instana.com/rancher/). + +This chart adds the Instana Agent to all schedulable nodes in your cluster via a `DaemonSet`. diff --git a/charts/instana/instana-agent/2.0.2/crds/operator_customresourcedefinition_agents_instana_io.yml b/charts/instana/instana-agent/2.0.2/crds/operator_customresourcedefinition_agents_instana_io.yml new file mode 100644 index 0000000000..3bb5d7b2d7 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/crds/operator_customresourcedefinition_agents_instana_io.yml @@ -0,0 +1,4451 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.14.0 + name: agents.instana.io +spec: + group: instana.io + names: + categories: + - monitoring + - openshift-optional + kind: InstanaAgent + listKind: InstanaAgentList + plural: agents + shortNames: + - ia + singular: agent + scope: Namespaced + versions: + - name: v1 + schema: + openAPIV3Schema: + description: InstanaAgent is the Schema for the agents API + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: InstanaAgentSpec defines the desired state of the Instana Agent + properties: + agent: + description: Agent deployment specific fields. + properties: + additionalBackends: + description: |- + These are additional backends the Instana agent will report to besides + the one configured via the `agent.endpointHost`, `agent.endpointPort` and `agent.key` setting. + items: + properties: + endpointHost: + type: string + endpointPort: + type: string + key: + type: string + required: + - endpointHost + - endpointPort + type: object + type: array + configuration_yaml: + description: Supply Agent configuration e.g. for configuring certain Sensors. + type: string + downloadKey: + description: |- + The DownloadKey, sometimes known as "sales key", that allows you to download software from Instana. It might be needed to + specify this in addition to the Key. + type: string + endpointHost: + description: EndpointHost is the hostname of the Instana server your agents will connect to. + type: string + endpointPort: + description: EndpointPort is the port number (as a String) of the Instana server your agents will connect to. + type: string + env: + additionalProperties: + type: string + description: |- + Use the `env` field to set additional environment variables for the Instana Agent, for example: + env: + INSTANA_AGENT_TAGS: dev + type: object + host: + description: Host sets a host path to be mounted as the Agent Maven repository (mainly for debugging or development purposes) + properties: + repository: + type: string + type: object + image: + description: Override the container image used for the Instana Agent pods. + properties: + digest: + description: |- + Digest (a.k.a. Image ID) of the agent container image. If specified, it has priority over `agent.image.tag`, + which will then be ignored. + type: string + name: + description: Name is the name of the container image of the Instana agent. + type: string + pullPolicy: + description: PullPolicy specifies when to pull the image container. + type: string + pullSecrets: + description: |- + PullSecrets allows you to override the default pull secret that is created when `agent.image.name` starts with + "containers.instana.io". Setting `agent.image.pullSecrets` prevents the creation of the default "containers-instana-io" secret. + items: + description: |- + LocalObjectReference contains enough information to let you locate the + referenced object inside the same namespace. + properties: + name: + default: "" + description: |- + Name of the referent. + This field is effectively required, but due to backwards compatibility is + allowed to be empty. Instances of this type with an empty value here are + almost certainly wrong. + TODO: Add other useful fields. apiVersion, kind, uid? + More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Drop `kubebuilder:default` when controller-gen doesn't need it https://github.com/kubernetes-sigs/kubebuilder/issues/3896. + type: string + type: object + x-kubernetes-map-type: atomic + type: array + tag: + description: Tag is the name of the agent container image; if `agent.image.digest` is specified, this property is ignored. + type: string + type: object + instanaMvnRepoFeaturesPath: + description: Sets the INSTANA_MVN_REPOSITORY_FEATURES_PATH environment variable + type: string + instanaMvnRepoSharedPath: + description: Sets the INSTANA_MVN_REPOSITORY_SHARED_PATH environment variable + type: string + instanaMvnRepoUrl: + description: |- + Override for the Maven repository URL when the Agent needs to connect to a locally provided Maven repository 'proxy' + Alternative to `Host` for referencing a different Maven repo. + type: string + key: + description: Key is the secret token which your agent uses to authenticate to Instana's servers. + type: string + keysSecret: + description: |- + Rather than specifying the Key and optionally the DownloadKey, you can "bring your + own secret" creating it in the namespace in which you install the `instana-agent` and + specify its name in the `KeysSecret` field. The secret you create must contain a field called `key` and optionally one + called `downloadKey`, which contain, respectively, the values you'd otherwise set in `.agent.key` and `agent.downloadKey`. + type: string + listenAddress: + description: |- + ListenAddress is the IP addresses the Agent HTTP server will listen on. Normally this will just be localhost (`127.0.0.1`), + the pod public IP and any container runtime bridge interfaces. Set `listenAddress: *` for making the Agent listen on all + network interfaces. + type: string + minReadySeconds: + description: The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available + type: integer + mode: + description: |- + Set agent mode, possible options are APM, INFRASTRUCTURE or AWS. KUBERNETES should not be used but instead enabled via + `kubernetes.deployment.enabled: true`. + type: string + pod: + description: Override Agent Pod specific settings such as annotations, labels and resources. + properties: + affinity: + description: |- + agent.pod.affinity are affinities to influence agent pod assignment. + https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node matches the corresponding matchExpressions; the + node(s) with the highest sum are the most preferred. + items: + description: |- + An empty preferred scheduling term matches all objects with implicit weight 0 + (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + properties: + preference: + description: A node selector term, associated with the corresponding weight. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to an update), the system + may or may not try to eventually evict the pod from its node. + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + items: + description: |- + A null or empty node selector term matches no objects. The requirements of + them are ANDed. + The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the anti-affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling anti-affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the anti-affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the anti-affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + annotations: + additionalProperties: + type: string + description: agent.pod.annotations are additional annotations to be added to the agent pods. + type: object + claims: + description: |- + Claims lists the names of resources, defined in spec.resourceClaims, + that are used by this container. + + + This is an alpha field and requires enabling the + DynamicResourceAllocation feature gate. + + + This field is immutable. It can only be set for containers. + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + properties: + name: + description: |- + Name must match the name of one entry in pod.spec.resourceClaims of + the Pod where this field is used. It makes that resource available + inside a container. + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + labels: + additionalProperties: + type: string + description: agent.pod.labels are additional labels to be added to the agent pods. + type: object + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Limits describes the maximum amount of compute resources allowed. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + nodeSelector: + additionalProperties: + type: string + type: object + priorityClassName: + description: |- + agent.pod.priorityClassName is the name of an existing PriorityClass that should be set on the agent pods + https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ + type: string + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Requests describes the minimum amount of compute resources required. + If Requests is omitted for a container, it defaults to Limits if that is explicitly specified, + otherwise to an implementation-defined value. Requests cannot exceed Limits. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + tolerations: + description: agent.pod.tolerations are tolerations to influence agent pod assignment. + items: + description: |- + The pod this Toleration is attached to tolerates any taint that matches + the triple using the matching operator . + properties: + effect: + description: |- + Effect indicates the taint effect to match. Empty means match all taint effects. + When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: |- + Key is the taint key that the toleration applies to. Empty means match all taint keys. + If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: |- + Operator represents a key's relationship to the value. + Valid operators are Exists and Equal. Defaults to Equal. + Exists is equivalent to wildcard for value, so that a pod can + tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: |- + TolerationSeconds represents the period of time the toleration (which must be + of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, + it is not set, which means tolerate the taint forever (do not evict). Zero and + negative values will be treated as 0 (evict immediately) by the system. + format: int64 + type: integer + value: + description: |- + Value is the taint value the toleration matches to. + If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + type: object + type: array + type: object + proxyHost: + description: proxyHost sets the INSTANA_AGENT_PROXY_HOST environment variable. + type: string + proxyPassword: + description: proxyPassword sets the INSTANA_AGENT_PROXY_PASSWORD environment variable. + type: string + proxyPort: + description: proxyPort sets the INSTANA_AGENT_PROXY_PORT environment variable. + type: string + proxyProtocol: + description: proxyProtocol sets the INSTANA_AGENT_PROXY_PROTOCOL environment variable. + type: string + proxyUseDNS: + description: proxyUseDNS sets the INSTANA_AGENT_PROXY_USE_DNS environment variable. + type: boolean + proxyUser: + description: proxyUser sets the INSTANA_AGENT_PROXY_USER environment variable. + type: string + redactKubernetesSecrets: + description: RedactKubernetesSecrets sets the INSTANA_KUBERNETES_REDACT_SECRETS environment variable. + type: string + serviceMesh: + description: ServiceMesh sets the ENABLE_AGENT_SOCKET environment variable. + properties: + enabled: + type: boolean + type: object + tls: + description: |- + TLS for end-to-end encryption between the Instana Agent and clients accessing the Agent. + The Instana Agent does not yet allow enforcing TLS encryption, enabling makes it possible for clients to 'opt-in'. + So TLS is only enabled on a connection when requested by the client. + properties: + certificate: + description: certificate (together with key) is the alternative to an existing Secret. Must be base64 encoded. + format: byte + type: string + key: + description: key (together with certificate) is the alternative to an existing Secret. Must be base64 encoded. + format: byte + type: string + secretName: + description: secretName is the name of the secret that has the relevant files. + type: string + type: object + updateStrategy: + description: Control how to update the Agent DaemonSet + properties: + rollingUpdate: + description: |- + Rolling update config params. Present only if type = "RollingUpdate". + --- + TODO: Update this to follow our convention for oneOf, whatever we decide it + to be. Same as Deployment `strategy.rollingUpdate`. + See https://github.com/kubernetes/kubernetes/issues/35345 + properties: + maxSurge: + anyOf: + - type: integer + - type: string + description: |- + The maximum number of nodes with an existing available DaemonSet pod that + can have an updated DaemonSet pod during during an update. + Value can be an absolute number (ex: 5) or a percentage of desired pods (ex: 10%). + This can not be 0 if MaxUnavailable is 0. + Absolute number is calculated from percentage by rounding up to a minimum of 1. + Default value is 0. + Example: when this is set to 30%, at most 30% of the total number of nodes + that should be running the daemon pod (i.e. status.desiredNumberScheduled) + can have their a new pod created before the old pod is marked as deleted. + The update starts by launching new pods on 30% of nodes. Once an updated + pod is available (Ready for at least minReadySeconds) the old DaemonSet pod + on that node is marked deleted. If the old pod becomes unavailable for any + reason (Ready transitions to false, is evicted, or is drained) an updated + pod is immediatedly created on that node without considering surge limits. + Allowing surge implies the possibility that the resources consumed by the + daemonset on any given node can double if the readiness check fails, and + so resource intensive daemonsets should take into account that they may + cause evictions during disruption. + x-kubernetes-int-or-string: true + maxUnavailable: + anyOf: + - type: integer + - type: string + description: |- + The maximum number of DaemonSet pods that can be unavailable during the + update. Value can be an absolute number (ex: 5) or a percentage of total + number of DaemonSet pods at the start of the update (ex: 10%). Absolute + number is calculated from percentage by rounding up. + This cannot be 0 if MaxSurge is 0 + Default value is 1. + Example: when this is set to 30%, at most 30% of the total number of nodes + that should be running the daemon pod (i.e. status.desiredNumberScheduled) + can have their pods stopped for an update at any given time. The update + starts by stopping at most 30% of those DaemonSet pods and then brings + up new DaemonSet pods in their place. Once the new pods are available, + it then proceeds onto other DaemonSet pods, thus ensuring that at least + 70% of original number of DaemonSet pods are available at all times during + the update. + x-kubernetes-int-or-string: true + type: object + type: + description: Type of daemon set update. Can be "RollingUpdate" or "OnDelete". Default is RollingUpdate. + type: string + type: object + required: + - endpointHost + - endpointPort + type: object + cluster: + description: |- + Name of the cluster, that will be assigned to this cluster in Instana. Either specifying the 'cluster.name' or 'zone.name' + is mandatory. + properties: + name: + type: string + type: object + k8s_sensor: + properties: + deployment: + properties: + enabled: + type: boolean + minReadySeconds: + description: The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available + type: integer + pod: + description: Override pod resource requirements for the Kubernetes Sensor pods. + properties: + affinity: + description: |- + agent.pod.affinity are affinities to influence agent pod assignment. + https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node matches the corresponding matchExpressions; the + node(s) with the highest sum are the most preferred. + items: + description: |- + An empty preferred scheduling term matches all objects with implicit weight 0 + (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + properties: + preference: + description: A node selector term, associated with the corresponding weight. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to an update), the system + may or may not try to eventually evict the pod from its node. + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + items: + description: |- + A null or empty node selector term matches no objects. The requirements of + them are ANDed. + The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the anti-affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling anti-affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the anti-affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the anti-affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + claims: + description: |- + Claims lists the names of resources, defined in spec.resourceClaims, + that are used by this container. + + + This is an alpha field and requires enabling the + DynamicResourceAllocation feature gate. + + + This field is immutable. It can only be set for containers. + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + properties: + name: + description: |- + Name must match the name of one entry in pod.spec.resourceClaims of + the Pod where this field is used. It makes that resource available + inside a container. + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Limits describes the maximum amount of compute resources allowed. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + nodeSelector: + additionalProperties: + type: string + type: object + priorityClassName: + type: string + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Requests describes the minimum amount of compute resources required. + If Requests is omitted for a container, it defaults to Limits if that is explicitly specified, + otherwise to an implementation-defined value. Requests cannot exceed Limits. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + tolerations: + description: agent.pod.tolerations are tolerations to influence agent pod assignment. + items: + description: |- + The pod this Toleration is attached to tolerates any taint that matches + the triple using the matching operator . + properties: + effect: + description: |- + Effect indicates the taint effect to match. Empty means match all taint effects. + When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: |- + Key is the taint key that the toleration applies to. Empty means match all taint keys. + If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: |- + Operator represents a key's relationship to the value. + Valid operators are Exists and Equal. Defaults to Equal. + Exists is equivalent to wildcard for value, so that a pod can + tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: |- + TolerationSeconds represents the period of time the toleration (which must be + of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, + it is not set, which means tolerate the taint forever (do not evict). Zero and + negative values will be treated as 0 (evict immediately) by the system. + format: int64 + type: integer + value: + description: |- + Value is the taint value the toleration matches to. + If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + type: object + type: array + type: object + replicas: + description: Specify the number of replicas for the Kubernetes Sensor. + type: integer + type: object + image: + properties: + digest: + description: |- + Digest (a.k.a. Image ID) of the agent container image. If specified, it has priority over `agent.image.tag`, + which will then be ignored. + type: string + name: + description: Name is the name of the container image of the Instana agent. + type: string + pullPolicy: + description: PullPolicy specifies when to pull the image container. + type: string + tag: + description: Tag is the name of the agent container image; if `agent.image.digest` is specified, this property is ignored. + type: string + type: object + podDisruptionBudget: + description: Toggles the PDB for the K8s Sensor + properties: + enabled: + type: boolean + type: object + type: object + kubernetes: + description: |- + Allows for installment of the Kubernetes Sensor as separate pod. Which allows for better tailored resource settings + (mainly memory) both for the Agent pods and the Kubernetes Sensor pod. + properties: + deployment: + properties: + enabled: + type: boolean + minReadySeconds: + description: The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available + type: integer + pod: + description: Override pod resource requirements for the Kubernetes Sensor pods. + properties: + affinity: + description: |- + agent.pod.affinity are affinities to influence agent pod assignment. + https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node matches the corresponding matchExpressions; the + node(s) with the highest sum are the most preferred. + items: + description: |- + An empty preferred scheduling term matches all objects with implicit weight 0 + (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + properties: + preference: + description: A node selector term, associated with the corresponding weight. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to an update), the system + may or may not try to eventually evict the pod from its node. + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + items: + description: |- + A null or empty node selector term matches no objects. The requirements of + them are ANDed. + The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the anti-affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling anti-affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the anti-affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the anti-affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + claims: + description: |- + Claims lists the names of resources, defined in spec.resourceClaims, + that are used by this container. + + + This is an alpha field and requires enabling the + DynamicResourceAllocation feature gate. + + + This field is immutable. It can only be set for containers. + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + properties: + name: + description: |- + Name must match the name of one entry in pod.spec.resourceClaims of + the Pod where this field is used. It makes that resource available + inside a container. + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Limits describes the maximum amount of compute resources allowed. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + nodeSelector: + additionalProperties: + type: string + type: object + priorityClassName: + type: string + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: |- + Requests describes the minimum amount of compute resources required. + If Requests is omitted for a container, it defaults to Limits if that is explicitly specified, + otherwise to an implementation-defined value. Requests cannot exceed Limits. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + type: object + tolerations: + description: agent.pod.tolerations are tolerations to influence agent pod assignment. + items: + description: |- + The pod this Toleration is attached to tolerates any taint that matches + the triple using the matching operator . + properties: + effect: + description: |- + Effect indicates the taint effect to match. Empty means match all taint effects. + When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: |- + Key is the taint key that the toleration applies to. Empty means match all taint keys. + If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: |- + Operator represents a key's relationship to the value. + Valid operators are Exists and Equal. Defaults to Equal. + Exists is equivalent to wildcard for value, so that a pod can + tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: |- + TolerationSeconds represents the period of time the toleration (which must be + of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, + it is not set, which means tolerate the taint forever (do not evict). Zero and + negative values will be treated as 0 (evict immediately) by the system. + format: int64 + type: integer + value: + description: |- + Value is the taint value the toleration matches to. + If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + type: object + type: array + type: object + replicas: + description: Specify the number of replicas for the Kubernetes Sensor. + type: integer + type: object + type: object + openshift: + description: |- + Set to `True` to indicate the Operator is being deployed in a OpenShift cluster. Provides a hint so that RBAC etc is + configured correctly. Will attempt to auto-detect if unset. + type: boolean + opentelemetry: + description: 'Enables the OpenTelemetry gRPC endpoint on the Agent. If true, it will also apply `service.create: true`.' + properties: + enabled: + type: boolean + grpc: + properties: + enabled: + type: boolean + type: object + http: + properties: + enabled: + type: boolean + type: object + type: object + pinnedChartVersion: + description: |- + Specifying the PinnedChartVersion allows for 'pinning' the Helm Chart used by the Operator for installing the Agent + DaemonSet. Normally the Operator will always install and update to the latest Helm Chart version. + The Operator will check and make sure no 'unsupported' Chart versions can be selected. + type: string + podSecurityPolicy: + description: 'Specify a PodSecurityPolicy for the Instana Agent Pods. If enabled requires `rbac.create: true`.' + properties: + enabled: + type: boolean + name: + type: string + type: object + prometheus: + description: 'Enables the Prometheus endpoint on the Agent. If true, it will also apply `service.create: true`.' + properties: + remoteWrite: + properties: + enabled: + type: boolean + type: object + type: object + rbac: + description: Specifies whether RBAC resources should be created. + properties: + create: + type: boolean + type: object + service: + description: |- + Specifies whether to create the instana-agent `Service` to expose within the cluster. The Service can then be used e.g. + for the Prometheus remote-write, OpenTelemetry GRCP endpoint and other APIs. + Note: Requires Kubernetes 1.17+, as it uses topologyKeys. + properties: + create: + type: boolean + type: object + serviceAccount: + description: Specifies whether a ServiceAccount should be created (default `true`), and possibly the name to use. + properties: + annotations: + additionalProperties: + type: string + type: object + create: + type: boolean + name: + type: string + type: object + serviceMesh: + properties: + enabled: + type: boolean + type: object + zone: + description: Name of the zone in which the host(s) will be displayed on the map. Optional, but then 'cluster.name' must be specified. + properties: + name: + type: string + type: object + zones: + description: Zones can be used to specify agents in multiple zones split across different nodes in the cluster + items: + properties: + affinity: + description: Affinity is a group of affinity scheduling rules. + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node matches the corresponding matchExpressions; the + node(s) with the highest sum are the most preferred. + items: + description: |- + An empty preferred scheduling term matches all objects with implicit weight 0 + (i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op). + properties: + preference: + description: A node selector term, associated with the corresponding weight. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to an update), the system + may or may not try to eventually evict the pod from its node. + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + items: + description: |- + A null or empty node selector term matches no objects. The requirements of + them are ANDed. + The TopologySelectorTerm type implements a subset of the NodeSelectorTerm. + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + description: A list of node selector requirements by node's fields. + items: + description: |- + A node selector requirement is a selector that contains values, a key, and an operator + that relates the key and values. + properties: + key: + description: The label key that the selector applies to. + type: string + operator: + description: |- + Represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt. + type: string + values: + description: |- + An array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. If the operator is Gt or Lt, the values + array must have a single element, which will be interpreted as an integer. + This array is replaced during a strategic merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + description: Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)). + properties: + preferredDuringSchedulingIgnoredDuringExecution: + description: |- + The scheduler will prefer to schedule pods to nodes that satisfy + the anti-affinity expressions specified by this field, but it may choose + a node that violates one or more of the expressions. The node that is + most preferred is the one with the greatest sum of weights, i.e. + for each node that meets all of the scheduling requirements (resource + request, requiredDuringScheduling anti-affinity expressions, etc.), + compute a sum by iterating through the elements of this field and adding + "weight" to the sum if the node has pods which matches the corresponding podAffinityTerm; the + node(s) with the highest sum are the most preferred. + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + weight: + description: |- + weight associated with matching the corresponding podAffinityTerm, + in the range 1-100. + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + description: |- + If the anti-affinity requirements specified by this field are not met at + scheduling time, the pod will not be scheduled onto the node. + If the anti-affinity requirements specified by this field cease to be met + at some point during pod execution (e.g. due to a pod label update), the + system may or may not try to eventually evict the pod from its node. + When there are multiple elements, the lists of nodes corresponding to each + podAffinityTerm are intersected, i.e. all terms must be satisfied. + items: + description: |- + Defines a set of pods (namely those matching the labelSelector + relative to the given namespace(s)) that this pod should be + co-located (affinity) or not co-located (anti-affinity) with, + where co-located is defined as running on a node whose value of + the label with key matches that of any node on which + a pod of the set of pods is running + properties: + labelSelector: + description: |- + A label query over a set of resources, in this case pods. + If it's null, this PodAffinityTerm matches with no Pods. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + description: |- + MatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both matchLabelKeys and labelSelector. + Also, matchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + description: |- + MismatchLabelKeys is a set of pod label keys to select which pods will + be taken into consideration. The keys are used to lookup values from the + incoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)` + to select the group of existing pods which pods will be taken into consideration + for the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming + pod labels will be ignored. The default value is empty. + The same key is forbidden to exist in both mismatchLabelKeys and labelSelector. + Also, mismatchLabelKeys cannot be set when labelSelector isn't set. + This is an alpha field and requires enabling MatchLabelKeysInPodAffinity feature gate. + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + description: |- + A label query over the set of namespaces that the term applies to. + The term is applied to the union of the namespaces selected by this field + and the ones listed in the namespaces field. + null selector and null or empty namespaces list means "this pod's namespace". + An empty selector ({}) matches all namespaces. + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + items: + description: |- + A label selector requirement is a selector that contains values, a key, and an operator that + relates the key and values. + properties: + key: + description: key is the label key that the selector applies to. + type: string + operator: + description: |- + operator represents a key's relationship to a set of values. + Valid operators are In, NotIn, Exists and DoesNotExist. + type: string + values: + description: |- + values is an array of string values. If the operator is In or NotIn, + the values array must be non-empty. If the operator is Exists or DoesNotExist, + the values array must be empty. This array is replaced during a strategic + merge patch. + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + description: |- + matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels + map is equivalent to an element of matchExpressions, whose key field is "key", the + operator is "In", and the values array contains only "value". The requirements are ANDed. + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + description: |- + namespaces specifies a static list of namespace names that the term applies to. + The term is applied to the union of the namespaces listed in this field + and the ones selected by namespaceSelector. + null or empty namespaces list and null namespaceSelector means "this pod's namespace". + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + description: |- + This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching + the labelSelector in the specified namespaces, where co-located is defined as running on a node + whose value of the label with key topologyKey matches that of any node on which any of the + selected pods is running. + Empty topologyKey is not allowed. + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + mode: + type: string + name: + type: string + tolerations: + items: + description: |- + The pod this Toleration is attached to tolerates any taint that matches + the triple using the matching operator . + properties: + effect: + description: |- + Effect indicates the taint effect to match. Empty means match all taint effects. + When specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: |- + Key is the taint key that the toleration applies to. Empty means match all taint keys. + If the key is empty, operator must be Exists; this combination means to match all values and all keys. + type: string + operator: + description: |- + Operator represents a key's relationship to the value. + Valid operators are Exists and Equal. Defaults to Equal. + Exists is equivalent to wildcard for value, so that a pod can + tolerate all taints of a particular category. + type: string + tolerationSeconds: + description: |- + TolerationSeconds represents the period of time the toleration (which must be + of effect NoExecute, otherwise this field is ignored) tolerates the taint. By default, + it is not set, which means tolerate the taint forever (do not evict). Zero and + negative values will be treated as 0 (evict immediately) by the system. + format: int64 + type: integer + value: + description: |- + Value is the taint value the toleration matches to. + If the operator is Exists, the value should be empty, otherwise just a regular string. + type: string + type: object + type: array + type: object + type: array + required: + - agent + type: object + status: + properties: + conditions: + items: + description: "Condition contains details for one aspect of the current state of this API Resource.\n---\nThis struct is intended for direct use as an array at the field path .status.conditions. For example,\n\n\n\ttype FooStatus struct{\n\t // Represents the observations of a foo's current state.\n\t // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\"\n\t // +patchMergeKey=type\n\t // +patchStrategy=merge\n\t // +listType=map\n\t // +listMapKey=type\n\t Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"`\n\n\n\t // other fields\n\t}" + properties: + lastTransitionTime: + description: |- + lastTransitionTime is the last time the condition transitioned from one status to another. + This should be when the underlying condition changed. If that is not known, then using the time when the API field changed is acceptable. + format: date-time + type: string + message: + description: |- + message is a human readable message indicating details about the transition. + This may be an empty string. + maxLength: 32768 + type: string + observedGeneration: + description: |- + observedGeneration represents the .metadata.generation that the condition was set based upon. + For instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date + with respect to the current state of the instance. + format: int64 + minimum: 0 + type: integer + reason: + description: |- + reason contains a programmatic identifier indicating the reason for the condition's last transition. + Producers of specific condition types may define expected values and meanings for this field, + and whether the values are considered a guaranteed API. + The value should be a CamelCase string. + This field may not be empty. + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + type: string + status: + description: status of the condition, one of True, False, Unknown. + enum: + - "True" + - "False" + - Unknown + type: string + type: + description: |- + type of condition in CamelCase or in foo.example.com/CamelCase. + --- + Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be + useful (see .node.status.conditions), the ability to deconflict is important. + The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + maxLength: 316 + pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$ + type: string + required: + - lastTransitionTime + - message + - reason + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + configmap: + description: ResourceInfo holds Name and UID to given object + properties: + name: + type: string + uid: + type: string + required: + - name + - uid + type: object + configsecret: + description: ResourceInfo holds Name and UID to given object + properties: + name: + type: string + uid: + type: string + required: + - name + - uid + type: object + daemonset: + description: ResourceInfo holds Name and UID to given object + properties: + name: + type: string + uid: + type: string + required: + - name + - uid + type: object + lastUpdate: + format: date-time + type: string + leadingAgentPod: + additionalProperties: + description: ResourceInfo holds Name and UID to given object + properties: + name: + type: string + uid: + type: string + required: + - name + - uid + type: object + type: object + observedGeneration: + format: int64 + minimum: 0 + type: integer + oldVersionsUpdated: + type: boolean + operatorVersion: + pattern: ^v?(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$ + type: string + reason: + type: string + status: + description: AgentOperatorState type representing the running state of the Agent Operator itself. + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/instana/instana-agent/2.0.2/kubernetes.deployment.enabled.png b/charts/instana/instana-agent/2.0.2/kubernetes.deployment.enabled.png new file mode 100644 index 0000000000000000000000000000000000000000..9358f6f1428da753b468891bb443da5b607c0c5c GIT binary patch literal 71985 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 deploy + cm >> ds + dcm >> deploy \ No newline at end of file diff --git a/charts/instana/instana-agent/2.0.2/questions.yml b/charts/instana/instana-agent/2.0.2/questions.yml new file mode 100644 index 0000000000..bf9d1a46d4 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/questions.yml @@ -0,0 +1,236 @@ +questions: +# Basic agent configuration +- variable: agent.key + label: agent.key + description: "Your Instana Agent key is the secret token which your agent uses to authenticate to Instana's servers" + type: string + required: true + group: "Agent Configuration" +- variable: agent.endpointHost + label: agent.endpointHost + description: "The hostname of the Instana server your agents will connect to. Defaults to ingress-red-saas.instana.io for US and ROW. If in Europe, please use ingress-blue-saas.instana.io" + type: string + required: true + default: "ingress-red-saas.instana.io" + group: "Agent Configuration" +- variable: zone.name + label: zone.name + description: "Custom zone that detected technologies will be assigned to" + type: string + required: true + group: "Agent Configuration" +# Advanced agent configuration +- variable: advancedAgentConfiguration + description: "Show advanced configuration for the Instana Agent" + label: Show advanced configuration + type: boolean + default: false + show_subquestion_if: true + group: "Advanced Agent Configuration" + subquestions: + - variable: agent.configuration_yaml + label: agent.configuration_yaml (Optional) + description: "Custom content for the agent configuration.yaml file in YAML format. Please use the 'Edit as YAML' feature in the Rancher UI for the best editing experience." + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.downloadKey + label: agent.downloadKey (Optional) + description: "Your Instana download key" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.endpointPort + label: agent.endpointPort + description: "The Agent backend port number (as a string) of the Instana server your agents will connect to" + type: string + required: true + default: "443" + group: "Advanced Agent Configuration" + - variable: agent.image.name + label: agent.image.name + description: "The name of the Instana Agent container image" + type: string + required: true + default: "instana/agent" + group: "Advanced Agent Configuration" + - variable: agent.image.tag + label: agent.image.tag + description: "The tag name of the Instana Agent container image" + type: string + required: true + default: "latest" + group: "Advanced Agent Configuration" + - variable: agent.image.pullPolicy + label: agent.image.pullPolicy + description: "Specifies when to pull the Instana Agent image container" + type: string + required: true + default: "Always" + group: "Advanced Agent Configuration" + - variable: agent.listenAddress + label: agent.listenAddress (Optional) + description: "The IP address the agent HTTP server will listen to, or '*' for all interfaces" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.mode + label: agent.mode (Optional) + description: "Agent mode. Possible options are: APM, INFRASTRUCTURE or AWS" + type: enum + options: + - "APM" + - "INFRASTRUCTURE" + - "AWS" + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.annotations + label: agent.pod.annotations (Optional) + description: "Additional annotations to be added to the agent pods in YAML format. Please use the 'Edit as YAML' feature in the Rancher UI for the best editing experience." + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.limits.cpu + label: agent.pod.limits.cpu + description: "CPU units allocation limits for the agent pods" + type: string + required: true + default: "1.5" + group: "Advanced Agent Configuration" + - variable: agent.pod.limits.memory + label: agent.pod.limits.memory + description: "Memory allocation limits in MiB for the agent pods" + type: int + required: true + default: 512 + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyHost + label: agent.pod.proxyHost (Optional) + description: "Hostname/address of a proxy. Sets the INSTANA_AGENT_PROXY_HOST environment variable" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyPort + label: agent.pod.proxyPort (Optional) + description: "Port of a proxy. Sets the INSTANA_AGENT_PROXY_PORT environment variable" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyProtocol + label: agent.pod.proxyProtocol (Optional) + description: "Proxy protocol. Sets the INSTANA_AGENT_PROXY_PROTOCOL environment variable. Supported proxy types are http, socks4, socks5" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyUser + label: agent.pod.proxyUser (Optional) + description: "Username of the proxy auth. Sets the INSTANA_AGENT_PROXY_USER environment variable" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyPassword + label: agent.pod.proxyPassword (Optional) + description: "Password of the proxy auth. Sets the INSTANA_AGENT_PROXY_PASSWORD environment variable" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.proxyUseDNS + label: agent.pod.proxyUseDNS. (Optional) + description: "Boolean if proxy also does DNS. Sets the INSTANA_AGENT_PROXY_USE_DNS environment variable" + type: enum + options: + - "true" + - "false" + required: false + group: "Advanced Agent Configuration" + - variable: agent.pod.requests.cpu + label: agent.pod.requests.cpu + description: "Requested CPU units allocation for the agent pods" + type: string + required: true + default: "0.5" + group: "Advanced Agent Configuration" + - variable: agent.pod.requests.memory + label: agent.pod.requests.memory + description: "Requested memory allocation in MiB for the agent pods" + type: int + required: true + default: 512 + group: "Advanced Agent Configuration" + - variable: agent.pod.tolerations + label: agent.pod.tolerations (Optional) + description: "Tolerations to influence agent pod assignment in YAML format. Please use the 'Edit as YAML' feature in the Rancher UI for the best editing experience." + type: string + required: false + group: "Advanced Agent Configuration" + - variable: agent.redactKubernetesSecrets + label: agent.redactKubernetesSecrets (Optional) + description: "Enable additional secrets redaction for selected Kubernetes resources" + type: boolean + required: false + default: false + group: "Advanced Agent Configuration" + - variable: cluster.name + label: cluster.name (Optional) + description: "The name that will be assigned to this cluster in Instana. See the 'Installing the Chart' section in the 'Detailed Descriptions' tab for more details" + type: string + required: false + group: "Advanced Agent Configuration" + - variable: leaderElector.image.name + label: leaderElector.image.name + description: "The name of the leader elector container image" + type: string + required: true + default: "instana/leader-elector" + group: "Advanced Agent Configuration" + - variable: leaderElector.image.tag + label: leaderElector.image.tag + description: "The tag name of the leader elector container image" + type: string + required: true + default: "0.5.4" + group: "Advanced Agent Configuration" + - variable: leaderElector.port + label: leaderElector.port + description: "The port on which the leader elector sidecar is exposed" + type: int + required: true + default: 42655 + group: "Advanced Agent Configuration" +- variable: podSecurityPolicy.enable + label: podSecurityPolicy.enable (Optional) + description: "Specifies whether a PodSecurityPolicy should be authorized for the Instana Agent pods. Requires `rbac.create` to also be `true`" + type: boolean + show_if: "rbac.create=true" + required: false + default: false + group: "Pod Security Policy Configuration" +- variable: podSecurityPolicy.name + label: podSecurityPolicy.name (Optional) + description: "The name of an existing PodSecurityPolicy you would like to authorize for the Instana Agent pods. If not set and `podSecurityPolicy.enable` is `true`, a PodSecurityPolicy will be created with a name generated using the fullname template" + type: string + show_if: "rbac.create=true&&podSecurityPolicy.enable=true" + required: false + group: "Pod Security Policy Configuration" +- variable: rbac.create + label: rbac.create + description: "Specifies whether RBAC resources should be created" + type: boolean + required: true + default: true + group: "RBAC Configuration" +- variable: serviceAccount.create + label: serviceAccount.create + description: "Specifies whether a ServiceAccount should be created" + type: boolean + required: true + default: true + show_subquestion_if: true + group: "RBAC Configuration" + subquestions: + - variable: serviceAccount.name + label: Name of the ServiceAccount (Optional) + description: "The name of the ServiceAccount to use. If not set and `serviceAccount.create` is true, a name is generated using the fullname template." + type: string + required: false + group: "RBAC Configuration" diff --git a/charts/instana/instana-agent/2.0.2/templates/NOTES.txt b/charts/instana/instana-agent/2.0.2/templates/NOTES.txt new file mode 100644 index 0000000000..8243ecd20d --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/NOTES.txt @@ -0,0 +1,73 @@ +{{- if (and (not (or .Values.agent.key .Values.agent.keysSecret )) (and (not .Values.zone.name) (not .Values.cluster.name))) }} +############################################################################## +#### ERROR: You did not specify your secret agent key. #### +#### ERROR: You also did not specify a zone or name for this cluster. #### +############################################################################## + +This agent deployment will be incomplete until you set your agent key and zone or name for this cluster: + + helm upgrade {{ .Release.Name }} --reuse-values \ + --repo https://agents.instana.io/helm \ + --set agent.key=$(YOUR_SECRET_AGENT_KEY) \ + --set zone.name=$(YOUR_ZONE_NAME) instana-agent + +Alternatively, you may specify a cluster name and the zone will be detected from availability zone information on the host: + + helm upgrade {{ .Release.Name }} --reuse-values \ + --repo https://agents.instana.io/helm \ + --set agent.key=$(YOUR_SECRET_AGENT_KEY) \ + --set cluster.name=$(YOUR_CLUSTER_NAME) instana-agent + +- YOUR_SECRET_AGENT_KEY can be obtained from the Management Portal section of your Instana installation. +- YOUR_ZONE_NAME should be the zone that detected technologies will be assigned to. +- YOUR_CLUSTER_NAME should be the custom name of your cluster. + +At least one of zone.name or cluster.name is required. This cluster will be reported with the name of the zone unless you specify a cluster name. + +{{- else if (and (not .Values.zone.name) (not .Values.cluster.name)) }} +############################################################################## +#### ERROR: You did not specify a zone or name for this cluster. #### +############################################################################## + +This agent deployment will be incomplete until you set a zone for this cluster: + + helm upgrade {{ .Release.Name }} --reuse-values \ + --repo https://agents.instana.io/helm \ + --set zone.name=$(YOUR_ZONE_NAME) instana-agent + +Alternatively, you may specify a cluster name and the zone will be detected from availability zone information on the host: + + helm upgrade {{ .Release.Name }} --reuse-values \ + --repo https://agents.instana.io/helm \ + --set cluster.name=$(YOUR_CLUSTER_NAME) instana-agent + +- YOUR_ZONE_NAME should be the zone that detected technologies will be assigned to. +- YOUR_CLUSTER_NAME should be the custom name of your cluster. + +At least one of zone.name or cluster.name is required. This cluster will be reported with the name of the zone unless you specify a cluster name. + +{{- else if not (or .Values.agent.key .Values.agent.keysSecret )}} +############################################################################## +#### ERROR: You did not specify your secret agent key. #### +############################################################################## + +This agent deployment will be incomplete until you set your agent key: + + helm upgrade {{ .Release.Name }} --reuse-values \ + --repo https://agents.instana.io/helm \ + --set agent.key=$(YOUR_SECRET_AGENT_KEY) instana-agent + +- YOUR_SECRET_AGENT_KEY can be obtained from the Management Portal section of your Instana installation. + +{{- else -}} +Ensure to run `oc adm policy add-scc-to-user privileged -z instana-agent -n instana-agent` if running on OCP, otherwise agent pods will not be scheduled correctly. + +It may take a few moments for the agents to fully deploy. You can see what agents are running by listing resources in the {{ .Release.Namespace }} namespace: + + kubectl get all -n {{ .Release.Namespace }} + +You can get the logs for all of the agents with `kubectl logs`: + + kubectl logs -l app.kubernetes.io/name={{ .Release.Name }} -n {{ .Release.Namespace }} -c instana-agent + +{{- end }} diff --git a/charts/instana/instana-agent/2.0.2/templates/agent.yml b/charts/instana/instana-agent/2.0.2/templates/agent.yml new file mode 100644 index 0000000000..3d9fbb0659 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/agent.yml @@ -0,0 +1,269 @@ +--- +apiVersion: instana.io/v1 +kind: InstanaAgent +metadata: + name: instana-agent + namespace: instana-agent +spec: + zone: + name: {{ .Values.zone.name }} +{{- if .Values.zones }} +{{- toYaml $.Values.zones | nindent 4 }} +{{- end }} + cluster: + name: {{ .Values.cluster.name }} + agent: +{{- if .Values.agent.mode }} + mode: {{ .Values.agent.mode }} +{{- end }} +{{- if .Values.agent.key }} + key: {{ .Values.agent.key }} +{{- end }} +{{- if .Values.agent.downloadKey }} + downloadKey: {{ .Values.agent.downloadKey }} +{{- end }} +{{- if .Values.agent.keysSecret }} + keysSecret: {{ .Values.agent.keysSecret }} +{{- end }} +{{- if .Values.agent.listenAddress }} + listenAddress: {{ .Values.agent.listenAddress }} +{{- end }} + endpointHost: {{ .Values.agent.endpointHost }} + {{- if eq (typeOf .Values.agent.endpointPort) "string" }} + endpointPort: {{ .Values.agent.endpointPort }} + {{- else }} + endpointPort: {{ .Values.agent.endpointPort | quote }} + {{- end }} +{{- if .Values.agent.additionalBackends }} + additionalBackends: +{{- range $.Values.agent.additionalBackends }} + - endpointHost: {{ .endpointHost }} + {{- if eq (typeOf .endpointPort) "string" }} + endpointPort: {{ .endpointPort }} + {{- else }} + endpointPort: {{ .endpointPort | quote }} + {{- end }} + {{- if .key }} + key: {{ .key }} + {{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.tls }} +{{- if or .Values.agent.tls.secretName (and .Values.agent.tls.certificate .Values.agent.tls.key) }} + tls: +{{- if .Values.agent.tls.secretName }} + secretName: {{ .Values.agent.tls.secretName }} +{{- end }} +{{- if .Values.agent.tls.certificate }} + certificate: {{ .Values.agent.tls.certificate }} +{{- end }} +{{- if .Values.agent.tls.key }} + key: {{ .Values.agent.tls.key }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.image }} +{{- if or .Values.agent.image.name .Values.agent.image.digest .Values.agent.image.tag .Values.agent.image.pullPolicy .Values.agent.image.pullSecrets }} + image: +{{- if .Values.agent.image.name }} + name: {{ .Values.agent.image.name }} +{{- end }} +{{- if .Values.agent.image.digest }} + digest: {{ .Values.agent.image.digest }} +{{- end }} +{{- if .Values.agent.image.tag }} + tag: {{ .Values.agent.image.tag }} +{{- end }} +{{- if .Values.agent.image.pullPolicy }} + pullPolicy: {{ .Values.agent.image.pullPolicy }} +{{- end }} +{{- if .Values.agent.image.pullSecrets }} + pullSecrets: +{{- toYaml $.Values.agent.image.pullSecrets | nindent 6 }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.minReadySeconds }} + minReadySeconds: {{ .Values.agent.minReadySeconds }} +{{- end }} +{{- if .Values.agent.updateStrategy }} + updateStrategy: +{{- if .Values.agent.updateStrategy.type }} + type: {{ .Values.agent.updateStrategy.type }} +{{- end }} +{{- if .Values.agent.updateStrategy.rollingUpdate }} +{{- if .Values.agent.updateStrategy.rollingUpdate.maxUnavailable }} + rollingUpdate: + maxUnavailable: {{ .Values.agent.updateStrategy.maxUnavailable }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.pod }} +{{- if or .Values.agent.pod.annotations .Values.agent.pod.labels .Values.agent.pod.tolerations .Values.agent.pod.affinity .Values.agent.pod.priorityClassName .Values.agent.pod.requests .Values.agent.pod.limits .Values.agent.pod.nodeSelector }} + pod: +{{- if .Values.agent.pod.annotations }} + annotations: + {{- toYaml $.Values.agent.pod.annotations | nindent 8 }} +{{- end }} +{{- if .Values.agent.pod.labels }} + labels: + {{- toYaml $.Values.agent.pod.labels | nindent 8 }} +{{- end }} +{{- if .Values.agent.pod.tolerations }} + tolerations: + {{- toYaml $.Values.agent.pod.tolerations | nindent 8 }} +{{- end }} +{{- if .Values.agent.pod.affinity }} + affinity: + {{- toYaml $.Values.agent.pod.affinity | nindent 8 }} +{{- end }} +{{- if .Values.agent.pod.priorityClassName }} + priorityClassName: {{ .Values.agent.pod.priorityClassName }} +{{- end }} +{{- if .Values.agent.pod.requests }} +{{- if or .Values.agent.pod.requests.memory .Values.agent.pod.requests.cpu }} + requests: +{{- if .Values.agent.pod.requests.memory }} + memory: {{ .Values.agent.pod.requests.memory }} +{{- end }} +{{- if .Values.agent.pod.requests.cpu }} + cpu: {{ .Values.agent.pod.requests.cpu | quote }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.pod.limits }} +{{- if or .Values.agent.pod.limits.memory .Values.agent.pod.limits.cpu }} + limits: +{{- if .Values.agent.pod.limits.memory }} + memory: {{ .Values.agent.pod.limits.memory }} +{{- end }} +{{- if .Values.agent.pod.limits.cpu }} + cpu: {{ .Values.agent.pod.limits.cpu | quote }} +{{- end }} +{{- if .Values.agent.pod.nodeSelector }} + nodeSelector: + {{- toYaml $.Values.agent.pod.nodeSelector | nindent 8 }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.proxyHost }} + proxyHost: {{ .Values.agent.proxyHost }} +{{- end }} +{{- if .Values.agent.proxyPort }} + proxyPort: {{ .Values.agent.proxyPort }} +{{- end }} +{{- if .Values.agent.proxyProtocol }} + proxyProtocol: {{ .Values.agent.proxyProtocol }} +{{- end }} +{{- if .Values.agent.proxyUser }} + proxyUser: {{ .Values.agent.proxyUser }} +{{- end }} +{{- if .Values.agent.proxyPassword }} + proxyPassword: {{ .Values.agent.proxyPassword }} +{{- end }} +{{- if .Values.agent.proxyUseDNS }} + proxyUseDNS: {{ .Values.agent.proxyUseDNS }} +{{- end }} +{{- if .Values.agent.env }} + env: +{{- range $key, $value := .Values.agent.env }} + {{- if eq (typeOf $value) "string" }} + {{ $key }}: {{ $value }} + {{- else }} + {{ $key }}: {{ $value | quote }} + {{- end }} +{{- end }} +{{- end }} +{{- if .Values.agent.configuration_yaml }} +{{ $configuration_yaml_string := .Values.agent.configuration_yaml }} + configuration_yaml: |- +{{ $configuration_yaml_string | indent 6}} +{{- end }} +{{- if and .Values.agent.host .Values.agent.host.repository }} + host: + repository: {{ .Values.agent.host.repository }} +{{- end }} +{{- if .Values.agent.serviceMesh}} +{{- if .Values.agent.serviceMesh.enabled }} + serviceMesh: + enabled: {{ .Values.agent.serviceMesh.enabled }} +{{- end }} +{{- end }} +{{- if .Values.opentelemetry }} +{{- if or .Values.opentelemetry.grpc .Values.opentelemetry.http }} + opentelemetry: +{{- if .Values.opentelemetry.grpc.enabled }} + grpc: + enabled: {{ .Values.opentelemetry.grpc.enabled }} +{{- end }} +{{- if .Values.opentelemetry.http.enabled }} + http: + enabled: {{ .Values.opentelemetry.http.enabled }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.prometheus }} +{{- if .Values.prometheus.remoteWrite }} +{{- if .Values.prometheus.remoteWrite.enabled }} + prometheus: + remoteWrite: + enabled: {{ .Values.prometheus.remoteWrite.enabled }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.serviceAccount }} +{{- if or .Values.serviceAccount.create .Values.serviceAccount.annotations }} + serviceAccount: +{{- if .Values.serviceAccount.create }} + create: {{ .Values.serviceAccount.create }} +{{- end }} +{{- if .Values.serviceAccount.name }} + name: {{ .Values.serviceAccount.name }} +{{- end }} +{{- if .Values.serviceAccount.annotations }} + annotations: {{ .Values.serviceAccount.annotations }} + {{- toYaml $.Values.serviceAccount.annotations | nindent 6 }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.podSecurityPolicy }} +{{- if or .Values.podSecurityPolicy.enable .Values.podSecurityPolicy.name }} + podSecurityPolicy: +{{- if .Values.podSecurityPolicy.enable }} + enable: {{ .Values.podSecurityPolicy.enable }} +{{- end }} +{{- if .Values.podSecurityPolicy.name }} + name: {{ .Values.podSecurityPolicy.name }} +{{- end }} +{{- end }} +{{- end }} +{{- if .Values.k8s_sensor }} +{{- if or .Values.k8s_sensor.image .Values.k8s_sensor.deployment .Values.k8s_sensor.podDisruptionBudget }} + k8s_sensor: +{{- if .Values.k8s_sensor.image }} + image: +{{- if .Values.k8s_sensor.image.name }} + name: {{ .Values.k8s_sensor.image.name }} +{{- end }} +{{- if .Values.k8s_sensor.image.digest }} + digest: {{ .Values.k8s_sensor.image.digest }} +{{- end }} +{{- if .Values.k8s_sensor.image.tag }} + tag: {{ .Values.k8s_sensor.image.tag }} +{{- end }} +{{- if .Values.k8s_sensor.image.pullPolicy }} + pullPolicy: {{ .Values.k8s_sensor.image.pullPolicy }} +{{- end }} +{{- if .Values.k8s_sensor.deployment }} + deployment: + {{- toYaml $.Values.k8s_sensor.deployment | nindent 6 }} +{{- end }} +{{- if .Values.k8s_sensor.podDisruptionBudget }} + podDisruptionBudget: + {{- toYaml $.Values.k8s_sensor.podDisruptionBudget | nindent 6 }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_leader-election-role.yml b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_leader-election-role.yml new file mode 100644 index 0000000000..2a79075c0c --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_leader-election-role.yml @@ -0,0 +1,27 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: leader-election-role +rules: + - apiGroups: + - "" + - coordination.k8s.io + resources: + - configmaps + - leases + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_manager-role.yml b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_manager-role.yml new file mode 100644 index 0000000000..8dd9132923 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrole_manager-role.yml @@ -0,0 +1,176 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: manager-role +rules: + - nonResourceURLs: + - /healthz + - /version + verbs: + - get + - apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - get + - list + - watch + - apiGroups: + - apps + resources: + - daemonsets + - deployments + verbs: + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - apps + resources: + - daemonsets + - deployments + - replicasets + - statefulsets + verbs: + - get + - list + - watch + - apiGroups: + - apps.openshift.io + resources: + - deploymentconfigs + verbs: + - get + - list + - watch + - apiGroups: + - autoscaling + resources: + - horizontalpodautoscalers + verbs: + - get + - list + - watch + - apiGroups: + - batch + resources: + - cronjobs + - jobs + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - configmaps + - endpoints + - events + - namespaces + - nodes + - persistentvolumeclaims + - persistentvolumes + - pods + - pods/log + - replicationcontrollers + - resourcequotas + - services + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - configmaps + - secrets + - serviceaccounts + - services + verbs: + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - extensions + resources: + - deployments + - ingresses + - replicasets + verbs: + - get + - list + - watch + - apiGroups: + - instana.io + resources: + - agents + verbs: + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - instana.io + resources: + - agents/finalizers + verbs: + - update + - apiGroups: + - instana.io + resources: + - agents/status + verbs: + - get + - patch + - update + - apiGroups: + - networking.k8s.io + resources: + - ingresses + verbs: + - get + - list + - watch + - apiGroups: + - policy + resourceNames: + - instana-agent-k8sensor + resources: + - podsecuritypolicies + verbs: + - use + - apiGroups: + - rbac.authorization.k8s.io + resources: + - clusterrolebindings + - clusterroles + verbs: + - bind + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_leader-election-rolebinding.yml b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_leader-election-rolebinding.yml new file mode 100644 index 0000000000..904203ce6f --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_leader-election-rolebinding.yml @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: leader-election-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: leader-election-role +subjects: + - kind: ServiceAccount + name: instana-agent-operator + namespace: instana-agent diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_manager-rolebinding.yml b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_manager-rolebinding.yml new file mode 100644 index 0000000000..b7609bd581 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_clusterrolebinding_manager-rolebinding.yml @@ -0,0 +1,13 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: manager-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: manager-role +subjects: + - kind: ServiceAccount + name: instana-agent-operator + namespace: instana-agent diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_configmap_manager-config.yml b/charts/instana/instana-agent/2.0.2/templates/operator_configmap_manager-config.yml new file mode 100644 index 0000000000..70ceeb82dc --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_configmap_manager-config.yml @@ -0,0 +1,17 @@ +--- +apiVersion: v1 +data: + controller_manager_config.yaml: | + apiVersion: controller-runtime.sigs.k8s.io/v1alpha1 + kind: ControllerManagerConfig + health: + healthProbeBindAddress: :8081 + metrics: + bindAddress: 127.0.0.1:8080 + leaderElection: + leaderElect: true + resourceName: 819a9291.instana.io +kind: ConfigMap +metadata: + name: manager-config + namespace: instana-agent diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_deployment_controller-manager.yml b/charts/instana/instana-agent/2.0.2/templates/operator_deployment_controller-manager.yml new file mode 100644 index 0000000000..4b34fc8598 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_deployment_controller-manager.yml @@ -0,0 +1,66 @@ +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/name: instana-agent-operator + name: controller-manager + namespace: instana-agent +spec: + replicas: 2 + selector: + matchLabels: + app.kubernetes.io/name: instana-agent-operator + template: + metadata: + labels: + app.kubernetes.io/name: instana-agent-operator + spec: + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: kubernetes.io/os + operator: In + values: + - linux + - matchExpressions: + - key: kubernetes.io/arch + operator: In + values: + - amd64 + - ppc64le + - s390x + - arm64 + containers: + - args: + - --leader-elect + command: + - /manager + image: icr.io/instana/instana-agent-operator:2.1.9 + imagePullPolicy: Always + livenessProbe: + httpGet: + path: /healthz + port: 8081 + initialDelaySeconds: 15 + periodSeconds: 20 + name: manager + readinessProbe: + httpGet: + path: /readyz + port: 8081 + initialDelaySeconds: 5 + periodSeconds: 10 + resources: + limits: + cpu: 200m + memory: 600Mi + requests: + cpu: 200m + memory: 200Mi + securityContext: + allowPrivilegeEscalation: true + serviceAccountName: instana-agent-operator + terminationGracePeriodSeconds: 10 diff --git a/charts/instana/instana-agent/2.0.2/templates/operator_serviceaccount_instana-agent-operator.yml b/charts/instana/instana-agent/2.0.2/templates/operator_serviceaccount_instana-agent-operator.yml new file mode 100644 index 0000000000..c23ec50a4a --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/templates/operator_serviceaccount_instana-agent-operator.yml @@ -0,0 +1,6 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: instana-agent-operator + namespace: instana-agent diff --git a/charts/instana/instana-agent/2.0.2/values.yaml b/charts/instana/instana-agent/2.0.2/values.yaml new file mode 100644 index 0000000000..9064973f68 --- /dev/null +++ b/charts/instana/instana-agent/2.0.2/values.yaml @@ -0,0 +1,271 @@ +# name is the value which will be used as the base resource name for various resources associated with the agent. +# name: instana-agent + +agent: + # agent.mode is used to set agent mode and it can be APM, INFRASTRUCTURE or AWS + # mode: APM + + # agent.key is the secret token which your agent uses to authenticate to Instana's servers. + key: null + # agent.downloadKey is key, sometimes known as "sales key", that allows you to download, + # software from Instana. + # downloadKey: null + + # Rather than specifying the agent key and optionally the download key, you can "bring your + # own secret" creating it in the namespace in which you install the `instana-agent` and + # specify its name in the `keysSecret` field. The secret you create must contains + # a field called `key` and optionally one called `downloadKey`, which contain, respectively, + # the values you'd otherwise set in `.agent.key` and `agent.downloadKey`. + # keysSecret: null + + # agent.listenAddress is the IP address the agent HTTP server will listen to. + # listenAddress: "*" + + # agent.endpointHost is the hostname of the Instana server your agents will connect to. + # endpointHost: ingress-red-saas.instana.io + # agent.endpointPort is the port number (as a String) of the Instana server your agents will connect to. + # endpointPort: 443 + + # These are additional backends the Instana agent will report to besides + # the one configured via the `agent.endpointHost`, `agent.endpointPort` and `agent.key` setting + # additionalBackends: [] + # - endpointHost: ingress.instana.io + # endpointPort: 443 + # key: + + # TLS for end-to-end encryption between Instana agent and clients accessing the agent. + # The Instana agent does not yet allow enforcing TLS encryption. + # TLS is only enabled on a connection when requested by the client. + # tls: + # In order to enable TLS, a secret of type kubernetes.io/tls must be specified. + # secretName is the name of the secret that has the relevant files. + # secretName: null + # Otherwise, the certificate and the private key must be provided as base64 encoded. + # certificate: null + # key: null + + # image: + # agent.image.name is the name of the container image of the Instana agent. + # name: icr.io/instana/agent + # agent.image.digest is the digest (a.k.a. Image ID) of the agent container image; if specified, it has priority over agent.image.tag, which will be ignored. + # digest: + # agent.image.tag is the tag name of the agent container image; if agent.image.digest is specified, this property is ignored. + # tag: latest + # agent.image.pullPolicy specifies when to pull the image container. + # pullPolicy: Always + # agent.image.pullSecrets allows you to override the default pull secret that is created when agent.image.name starts with "containers.instana.io" + # Setting agent.image.pullSecrets prevents the creation of the default "containers-instana-io" secret. + # pullSecrets: + # - name: my_awesome_secret_instead + # If you want no imagePullSecrets to be specified in the agent pod, you can just pass an empty array to agent.image.pullSecrets + # pullSecrets: [] + + # The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available + # minReadySeconds: 0 + + # updateStrategy: + # type: RollingUpdate + # rollingUpdate: + # maxUnavailable: 1 + + # pod: + # agent.pod.annotations are additional annotations to be added to the agent pods. + # annotations: {} + + # agent.pod.labels are additional labels to be added to the agent pods. + # labels: {} + + # agent.pod.tolerations are tolerations to influence agent pod assignment. + # https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + # tolerations: [] + + # agent.pod.affinity are affinities to influence agent pod assignment. + # https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + # affinity: {} + + # agent.pod.priorityClassName is the name of an existing PriorityClass that should be set on the agent pods + # https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ + # priorityClassName: null + + # agent.pod.nodeSelector are selectors to influence where agent pods should be scheduled. + # nodeSelector: + # location: 'us-central1-c' + #nodeSelector: null + + # agent.pod.requests and agent.pod.limits adjusts the resource assignments for the DaemonSet agent + # regardless of the kubernetes.deployment.enabled setting + # requests: + # agent.pod.requests.memory is the requested memory allocation in MiB for the agent pods. + # memory: 768Mi + # agent.pod.requests.cpu are the requested CPU units allocation for the agent pods. + # cpu: 0.5 + # limits: + # agent.pod.limits.memory set the memory allocation limits in MiB for the agent pods. + # memory: 768Mi + # agent.pod.limits.cpu sets the CPU units allocation limits for the agent pods. + # cpu: 1.5 + + # agent.proxyHost sets the INSTANA_AGENT_PROXY_HOST environment variable. + # proxyHost: null + # agent.proxyPort sets the INSTANA_AGENT_PROXY_PORT environment variable. + # proxyPort: 80 + # agent.proxyProtocol sets the INSTANA_AGENT_PROXY_PROTOCOL environment variable. + # proxyProtocol: HTTP + # agent.proxyUser sets the INSTANA_AGENT_PROXY_USER environment variable. + # proxyUser: null + # agent.proxyPassword sets the INSTANA_AGENT_PROXY_PASSWORD environment variable. + # proxyPassword: null + # agent.proxyUseDNS sets the INSTANA_AGENT_PROXY_USE_DNS environment variable. + # proxyUseDNS: false + + # use this to set additional environment variables for the instana agent + # for example: + # env: + # INSTANA_AGENT_TAGS: dev + # env: {} + + configuration_yaml: | + # Manual a-priori configuration. Configuration will be only used when the sensor + # is actually installed by the agent. + # The commented out example values represent example configuration and are not + # necessarily defaults. Defaults are usually 'absent' or mentioned separately. + # Changes are hot reloaded unless otherwise mentioned. + + # It is possible to create files called 'configuration-abc.yaml' which are + # merged with this file in file system order. So 'configuration-cde.yaml' comes + # after 'configuration-abc.yaml'. Only nested structures are merged, values are + # overwritten by subsequent configurations. + + # Secrets + # To filter sensitive data from collection by the agent, all sensors respect + # the following secrets configuration. If a key collected by a sensor matches + # an entry from the list, the value is redacted. + #com.instana.secrets: + # matcher: 'contains-ignore-case' # 'contains-ignore-case', 'contains', 'regex' + # list: + # - 'key' + # - 'password' + # - 'secret' + + # Host + #com.instana.plugin.host: + # tags: + # - 'dev' + # - 'app1' + + # agent.redactKubernetesSecrets sets the INSTANA_KUBERNETES_REDACT_SECRETS environment variable. + # redactKubernetesSecrets: null + + # agent.host.repository sets a host path to be mounted as the agent maven repository (for debugging or development purposes) + host: + repository: null + + # agent.serviceMesh.enabled sets the ENABLE_AGENT_SOCKET environment variable. + serviceMesh: + # enabled: true + +cluster: + # cluster.name represents the name that will be assigned to this cluster in Instana + name: null + +# openshift specifies whether the cluster role should include openshift permissions and other tweaks to the YAML. +# The chart will try to auto-detect if the cluster is OpenShift, so you will likely not even need to set this explicitly. +# openshift: true + +# rbac: + # Specifies whether RBAC resources should be created + # create: true + +# opentelemetry: + # enabled: false # legacy setting, will only enable grpc, defaults to false + # grpc: + # enabled: true # takes precedence over legacy settings above, defaults to true if "grpc:" is present + # http: + # enabled: true # allows to enable http endpoints, defaults to true if "http:" is present + +# prometheus: +# remoteWrite: +# enabled: false # If true, it will also apply `service.create=true` + +# serviceAccount: + # Specifies whether a ServiceAccount should be created + # create: true + # The name of the ServiceAccount to use. + # If not set and `create` is true, a name is generated using the fullname template + # name: instana-agent + # Annotations to add to the service account + # annotations: {} + +# podSecurityPolicy: + # Specifies whether a PodSecurityPolicy should be authorized for the Instana Agent pods. + # Requires `rbac.create` to be `true` as well and K8s version below v1.25. + # enable: false + # The name of an existing PodSecurityPolicy you would like to authorize for the Instana Agent pods. + # If not set and `enable` is true, a PodSecurityPolicy will be created with a name generated using the fullname template. + # name: null + +zone: + # zone.name is the custom zone that detected technologies will be assigned to + name: null + +# k8s_sensor: +# image: + # k8s_sensor.image.name is the name of the container image of the Instana agent. + # name: icr.io/instana/k8sensor + # k8s_sensor.image.digest is the digest (a.k.a. Image ID) of the agent container image; if specified, it has priority over agent.image.tag, which will be ignored. + #digest: + # k8s_sensor.image.tag is the tag name of the agent container image; if agent.image.digest is specified, this property is ignored. + # tag: latest + # k8s_sensor.image.pullPolicy specifies when to pull the image container. + # pullPolicy: Always + # deployment: + # Specifies whether or not to enable the Deployment and turn off the Kubernetes sensor in the DaemonSet + # enabled: true + # Use three replicas to ensure the HA by the default. + # replicas: 3 + # k8s_sensor.deployment.pod adjusts the resource assignments for the agent independently of the DaemonSet agent when k8s_sensor.deployment.enabled=true + # pod: + # requests: + # k8s_sensor.deployment.pod.requests.memory is the requested memory allocation in MiB for the agent pods. + # memory: 128Mi + # k8s_sensor.deployment.pod.requests.cpu are the requested CPU units allocation for the agent pods. + # cpu: 120m + # limits: + # k8s_sensor.deployment.pod.limits.memory set the memory allocation limits in MiB for the agent pods. + # memory: 2048Mi + # k8s_sensor.deployment.pod.limits.cpu sets the CPU units allocation limits for the agent pods. + # cpu: 500m + # affinity: + # podAntiAffinity: + # Soft anti-affinity policy: try not to schedule multiple kubernetes-sensor pods on the same node. + # If the policy is set to "requiredDuringSchedulingIgnoredDuringExecution", if the cluster has + # fewer nodes than the amount of desired replicas, `helm install/upgrade --wait` will not return. + # preferredDuringSchedulingIgnoredDuringExecution: + # - weight: 100 + # podAffinityTerm: + # labelSelector: + # matchExpressions: + # - key: instana/agent-mode + # operator: In + # values: [ KUBERNETES ] + # topologyKey: "kubernetes.io/hostname" + # The minimum number of seconds for which a newly created Pod should be ready without any of its containers crashing, for it to be considered available + # minReadySeconds: 0 + # podDisruptionBudget: + # Specifies whether or not to setup a pod disruption budget for the k8sensor deployment + # enabled: false + +# zones: +# # Configure use of zones to use tolerations as the basis to associate a specific daemonset per tainted node pool +# - name: pool-01 +# tolerations: +# - key: "pool" +# operator: "Equal" +# value: "pool-01" +# effect: "NoExecute" +# - name: pool-02 +# tolerations: +# - key: "pool" +# operator: "Equal" +# value: "pool-02" +# effect: "NoExecute" \ No newline at end of file diff --git a/charts/percona/psmdb-db/1.18.0/.helmignore b/charts/percona/psmdb-db/1.18.0/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/percona/psmdb-db/1.18.0/Chart.yaml b/charts/percona/psmdb-db/1.18.0/Chart.yaml new file mode 100644 index 0000000000..4dfedbc8ac --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/Chart.yaml @@ -0,0 +1,19 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Percona Server for MongoDB + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: psmdb-db +apiVersion: v2 +appVersion: 1.18.0 +description: A Helm chart for installing Percona Server MongoDB Cluster Databases + using the PSMDB Operator. +home: https://www.percona.com/doc/kubernetes-operator-for-psmongodb/index.html +icon: file://assets/icons/psmdb-db.png +kubeVersion: '>=1.21-0' +maintainers: +- email: tomislav.plavcic@percona.com + name: tplavcic +- email: natalia.marukovich@percona.com + name: nmarukovich +name: psmdb-db +version: 1.18.0 diff --git a/charts/percona/psmdb-db/1.18.0/README.md b/charts/percona/psmdb-db/1.18.0/README.md new file mode 100644 index 0000000000..6b7671d2ac --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/README.md @@ -0,0 +1,316 @@ +# Percona Server for MongoDB + +This chart deploys Percona Server for MongoDB Cluster on Kubernetes controlled by Percona Operator for MongoDB. + +Useful links: + +- [Operator Github repository](https://github.com/percona/percona-server-mongodb-operator) +- [Operator Documentation](https://www.percona.com/doc/kubernetes-operator-for-psmongodb/index.html) + +## Pre-requisites + +- Percona Operator for MongoDB running in your Kubernetes cluster. See installation details [here](https://github.com/percona/percona-helm-charts/blob/main/charts/psmdb-operator) or in the [Operator Documentation](https://www.percona.com/doc/kubernetes-operator-for-psmongodb/helm.html). +- Kubernetes 1.28+ +- Helm v3 + +# Chart Details + +This chart will deploy Percona Server for MongoDB Cluster in Kubernetes. It will create a Custom Resource, and the Operator will trigger the creation of corresponding Kubernetes primitives: StatefulSets, Pods, Secrets, etc. + +## Installing the Chart + +To install the chart with the `psmdb` release name using a dedicated namespace (recommended): + +```sh +helm repo add percona https://percona.github.io/percona-helm-charts/ +helm install my-db percona/psmdb-db --version 1.18.0 --namespace my-namespace +``` + +The chart can be customized using the following configurable parameters: + +| Parameter | Description | Default | +| --------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------- | +| `crVersion` | CR Cluster Manifest version | `1.18.0` | +| `pause` | Stop PSMDB Database safely | `false` | +| `unmanaged` | Start cluster and don't manage it (cross cluster replication) | `false` | +| `enableVolumeExpansion` | Allows to resize `PersistentVolumeClaim`s by changing `.volumeSpec.persistentVolumeClaim.resources` field | `false` | +| `unsafeFlags.tls` | Allows users from configuring a cluster without TLS/SSL certificates | `false` | +| `unsafeFlags.replsetSize` | Allows users from configuring a cluster with unsafe parameters: starting it with less than 3 replica set instances or with an even number of replica set instances without additional arbiter | `false` | +| `unsafeFlags.mongosSize` | Allows users from configuring a sharded cluster with less than 3 config server Pods or less than 2 mongos Pods | `false` | +| `unsafeFlags.terminationGracePeriod` | Allows users from configuring a sharded cluster without termination grace period for replica set | `false` | +| `unsafeFlags.backupIfUnhealthy` | Allows running backup on a cluster with failed health checks | `false` | +| `clusterServiceDNSSuffix` | The (non-standard) cluster domain to be used as a suffix of the Service name | `""` | +| `clusterServiceDNSMode` | Mode for the cluster service dns (Internal/ServiceMesh) | `""` | +| `annotations` | PSMDB custom resource annotations | `{}` | +| `ignoreAnnotations` | The list of annotations to be ignored by the Operator | `[]` | +| `ignoreLabels` | The list of labels to be ignored by the Operator | `[]` | +| `multiCluster.enabled` | Enable Multi Cluster Services (MCS) cluster mode | `false` | +| `multiCluster.DNSSuffix` | The cluster domain to be used as a suffix for multi-cluster Services used by Kubernetes | `""` | +| `updateStrategy` | Regulates the way how PSMDB Cluster Pods will be updated after setting a new image | `SmartUpdate` | +| `upgradeOptions.versionServiceEndpoint` | Endpoint for actual PSMDB Versions provider | `https://check.percona.com/versions/` | +| `upgradeOptions.apply` | PSMDB image to apply from version service - recommended, latest, actual version like 4.4.2-4 | `disabled` | +| `upgradeOptions.schedule` | Cron formatted time to execute the update | `"0 2 * * *"` | +| `upgradeOptions.setFCV` | Set feature compatibility version on major upgrade | `false` | +| `finalizers:percona.com/delete-psmdb-pvc` | Set this if you want to delete database persistent volumes on cluster deletion | `[]` | +| `finalizers:percona.com/delete-psmdb-pods-in-order` | Set this if you want to delete PSMDB pods in order (primary last) | `[]` | +| `finalizers:percona.com/delete-pitr-chunks` | Set this if you want to delete all pitr chunks on cluster deletion | `[]` | +| `image.repository` | PSMDB Container image repository | `percona/percona-server-mongodb` | +| `image.tag` | PSMDB Container image tag | `7.0.14-8-multi` | +| `imagePullPolicy` | The policy used to update images | `Always` | +| `imagePullSecrets` | PSMDB Container pull secret | `[]` | +| `initImage.repository` | Repository for custom init image | `""` | +| `initImage.tag` | Tag for custom init image | `""` | +| `initContainerSecurityContext` | A custom Kubernetes Security Context for a Container for the initImage | `{}` | +| | +| `tls.mode` | Control usage of TLS (allowTLS, preferTLS, requireTLS, disabled) | `preferTLS` | +| `tls.certValidityDuration` | The validity duration of the external certificate for cert manager | `""` | +| `tls.allowInvalidCertificates` | If enabled the mongo shell will not attempt to validate the server certificates | `true` | +| `tls.issuerConf.name` | A cert-manager issuer name | `""` | +| `tls.issuerConf.kind` | A cert-manager issuer kind | `""` | +| `tls.issuerConf.group` | A cert-manager issuer group | `""` | +| | +| `secrets.users` | The name of the Secrets object for the MongoDB users required to run the operator | `""` | +| `secrets.encryptionKey` | Set secret for data at rest encryption key | `""` | +| `secrets.keyFile` | Specifies a secret key file for authenticating MongoDB instances | `""` | +| `secrets.vault` | Specifies a secret object to provide integration with HashiCorp Vault | `""` | +| `secrets.ldapSecret` | Specifies a secret object for LDAP over TLS connection between MongoDB and OpenLDAP server | `""` | +| `secrets.sse` | The name of the Secrets object for server side encryption credentials | `""` | +| `secrets.ssl` | A secret with TLS certificate generated for external communications | `""` | +| `secrets.sslInternal` | A secret with TLS certificate generated for internal communications | `""` | +| | +| `pmm.enabled` | Enable integration with [Percona Monitoring and Management software](https://www.percona.com/blog/2020/07/23/using-percona-kubernetes-operators-with-percona-monitoring-and-management/) | `false` | +| `pmm.image.repository` | PMM Container image repository | `percona/pmm-client` | +| `pmm.image.tag` | PMM Container image tag | `2.43.2` | +| `pmm.serverHost` | PMM server related K8S service hostname | `monitoring-service` | +| `pmm.containerSecurityContext` | Set the security context for PMM container | `{}` | +| `pmm.resources` | Set resources for PMM container | `{}` | +| `pmm.mongodParams` | PMM mongod params | `""` | +| `pmm.mongosParams` | PMM mongos params | `""` | +| | +| `replsets.rs0.name` | ReplicaSet name | `rs0` | +| `replsets.rs0.size` | ReplicaSet size (pod quantity) | `3` | +| `replsets.rs0.terminationGracePeriodSeconds` | The amount of seconds Kubernetes will wait for a clean replica set Pods termination | `""` | +| `replsets.rs0.externalNodes.host` | The URL or IP address of the external replset instance | `""` | +| `replsets.rs0.externalNodes.port` | The port number of the external replset instance | `""` | +| `replsets.rs0.externalNodes.votes` | The number of [votes](https://docs.mongodb.com/manual/reference/replica-configuration/#mongodb-rsconf-rsconf.members-n-.votes) of the external replset instance | `""` | +| `replsets.rs0.externalNodes.priority` | The [priority](https://docs.mongodb.com/manual/reference/replica-configuration/#mongodb-rsconf-rsconf.members-n-.priority) of the external replset instance | `""` | +| `replsets.rs0.configuration` | Custom config for mongod in replica set | `""` | +| `replsets.rs0.topologySpreadConstraints` | Control how Pods are spread across your cluster among failure-domains such as regions, zones, nodes, and other user-defined topology domains | `{}` | +| `replsets.rs0.replsetOverrides` | Use if you need tooverride the replica set members FQDNs with custom hostnames. Each key under `replsetOverrides` should be name of a Pod. The Operator won’t perform any validation for hostnames, so it's the user’s responsibility to ensure connectivity | `{}` | +| `replsets.rs0.serviceAccountName` | Run replicaset Containers under specified K8S SA | `""` | +| `replsets.rs0.affinity.antiAffinityTopologyKey` | ReplicaSet Pod affinity | `kubernetes.io/hostname` | +| `replsets.rs0.affinity.advanced` | ReplicaSet Pod advanced affinity | `{}` | +| `replsets.rs0.tolerations` | ReplicaSet Pod tolerations | `[]` | +| `replsets.rs0.priorityClass` | ReplicaSet Pod priorityClassName | `""` | +| `replsets.rs0.annotations` | ReplicaSet Pod annotations | `{}` | +| `replsets.rs0.labels` | ReplicaSet Pod labels | `{}` | +| `replsets.rs0.nodeSelector` | ReplicaSet Pod nodeSelector labels | `{}` | +| `replsets.rs0.livenessProbe` | ReplicaSet Pod livenessProbe structure | `{}` | +| `replsets.rs0.readinessProbe` | ReplicaSet Pod readinessProbe structure | `{}` | +| `replsets.rs0.storage` | Set cacheSizeRatio or other custom MongoDB storage options | `{}` | +| `replsets.rs0.podSecurityContext` | Set the security context for a Pod | `{}` | +| `replsets.rs0.containerSecurityContext` | Set the security context for a Container | `{}` | +| `replsets.rs0.runtimeClass` | ReplicaSet Pod runtimeClassName | `""` | +| `replsets.rs0.sidecars` | ReplicaSet Pod sidecars | `{}` | +| `replsets.rs0.sidecarVolumes` | ReplicaSet Pod sidecar volumes | `[]` | +| `replsets.rs0.sidecarPVCs` | ReplicaSet Pod sidecar PVCs | `[]` | +| `replsets.rs0.podDisruptionBudget.maxUnavailable` | ReplicaSet failed Pods maximum quantity | `1` | +| `replsets.rs0.splitHorizons` | External URI for Split-horizon for replica set Pods of the exposed cluster | `{}` | +| `replsets.rs0.expose.enabled` | Allow access to replicaSet from outside of Kubernetes | `false` | +| `replsets.rs0.expose.type` | Network service access point type | `ClusterIP` | +| `replsets.rs0.expose.loadBalancerIP` | Set client IP to Load Balancer | `""` | +| `replsets.rs0.expose.loadBalancerSourceRanges` | Limit client IP's access to Load Balancer | `{}` | +| `replsets.rs0.expose.annotations` | ReplicaSet service annotations | `{}` | +| `replsets.rs0.expose.labels` | ReplicaSet service labels | `{}` | +| `replsets.rs0.expose.internalTrafficPolicy` | ReplicaSet service internal traffic policy | `Local` | +| `replsets.rs0.expose.externalTrafficPolicy` | ReplicaSet service external traffic policy | `Local` | +| `replsets.rs0.schedulerName` | ReplicaSet Pod schedulerName | `""` | +| `replsets.rs0.resources` | ReplicaSet Pods resource requests and limits | `{}` | +| `replsets.rs0.volumeSpec` | ReplicaSet Pods storage resources | `{}` | +| `replsets.rs0.volumeSpec.emptyDir` | ReplicaSet Pods emptyDir K8S storage | `{}` | +| `replsets.rs0.volumeSpec.hostPath` | ReplicaSet Pods hostPath K8S storage | | +| `replsets.rs0.volumeSpec.hostPath.path` | ReplicaSet Pods hostPath K8S storage path | `""` | +| `replsets.rs0.volumeSpec.hostPath.type` | Type for hostPath volume | `Directory` | +| `replsets.rs0.volumeSpec.pvc` | ReplicaSet Pods PVC request parameters | | +| `replsets.rs0.volumeSpec.pvc.annotations` | The Kubernetes annotations metadata for Persistent Volume Claim | `{}` | +| `replsets.rs0.volumeSpec.pvc.labels` | The Kubernetes labels metadata for Persistent Volume Claim | `{}` | +| `replsets.rs0.volumeSpec.pvc.storageClassName` | ReplicaSet Pods PVC target storageClass | `""` | +| `replsets.rs0.volumeSpec.pvc.accessModes` | ReplicaSet Pods PVC access policy | `[]` | +| `replsets.rs0.volumeSpec.pvc.resources.requests.storage` | ReplicaSet Pods PVC storage size | `3Gi` | +| `replsets.rs0.hostAliases` | The IP address for Kubernetes host aliases | `[]` | +| `replsets.rs0.nonvoting.enabled` | Add MongoDB nonvoting Pods | `false` | +| `replsets.rs0.nonvoting.podSecurityContext` | Set the security context for a Pod | `{}` | +| `replsets.rs0.nonvoting.containerSecurityContext` | Set the security context for a Container | `{}` | +| `replsets.rs0.nonvoting.size` | Number of nonvoting Pods | `1` | +| `replsets.rs0.nonvoting.configuration` | Custom config for mongod nonvoting member | `""` | +| `replsets.rs0.nonvoting.serviceAccountName` | Run replicaset nonvoting Container under specified K8S SA | `""` | +| `replsets.rs0.nonvoting.affinity.antiAffinityTopologyKey` | Nonvoting Pods affinity | `kubernetes.io/hostname` | +| `replsets.rs0.nonvoting.affinity.advanced` | Nonvoting Pods advanced affinity | `{}` | +| `replsets.rs0.nonvoting.tolerations` | Nonvoting Pod tolerations | `[]` | +| `replsets.rs0.nonvoting.priorityClass` | Nonvoting Pod priorityClassName | `""` | +| `replsets.rs0.primaryPreferTagSelector.region` | Makes MongoDB instance to be selected as Primary based on specified region | `""` | +| `replsets.rs0.primaryPreferTagSelector.zone` | Makes MongoDB instance to be selected as Primary based on specified zone | `""` | +| `replsets.rs0.nonvoting.annotations` | Nonvoting Pod annotations | `{}` | +| `replsets.rs0.nonvoting.labels` | Nonvoting Pod labels | `{}` | +| `replsets.rs0.nonvoting.nodeSelector` | Nonvoting Pod nodeSelector labels | `{}` | +| `replsets.rs0.nonvoting.podDisruptionBudget.maxUnavailable` | Nonvoting failed Pods maximum quantity | `1` | +| `replsets.rs0.nonvoting.resources` | Nonvoting Pods resource requests and limits | `{}` | +| `replsets.rs0.nonvoting.volumeSpec` | Nonvoting Pods storage resources | `{}` | +| `replsets.rs0.nonvoting.volumeSpec.emptyDir` | Nonvoting Pods emptyDir K8S storage | `{}` | +| `replsets.rs0.nonvoting.volumeSpec.hostPath` | Nonvoting Pods hostPath K8S storage | | +| `replsets.rs0.nonvoting.volumeSpec.hostPath.path` | Nonvoting Pods hostPath K8S storage path | `""` | +| `replsets.rs0.nonvoting.volumeSpec.hostPath.type` | Type for hostPath volume | `Directory` | +| `replsets.rs0.nonvoting.volumeSpec.pvc` | Nonvoting Pods PVC request parameters | | +| `replsets.rs0.nonvoting.volumeSpec.pvc.annotations` | The Kubernetes annotations metadata for Persistent Volume Claim | `{}` | +| `replsets.rs0.nonvoting.volumeSpec.pvc.labels` | The Kubernetes labels metadata for Persistent Volume Claim | `{}` | +| `replsets.rs0.nonvoting.volumeSpec.pvc.storageClassName` | Nonvoting Pods PVC target storageClass | `""` | +| `replsets.rs0.nonvoting.volumeSpec.pvc.accessModes` | Nonvoting Pods PVC access policy | `[]` | +| `replsets.rs0.nonvoting.volumeSpec.pvc.resources.requests.storage` | Nonvoting Pods PVC storage size | `3Gi` | +| `replsets.rs0.arbiter.enabled` | Create MongoDB arbiter service | `false` | +| `replsets.rs0.arbiter.size` | MongoDB arbiter Pod quantity | `1` | +| `replsets.rs0.arbiter.serviceAccountName` | Run replicaset arbiter Container under specified K8S SA | `""` | +| `replsets.rs0.arbiter.affinity.antiAffinityTopologyKey` | MongoDB arbiter Pod affinity | `kubernetes.io/hostname` | +| `replsets.rs0.arbiter.affinity.advanced` | MongoDB arbiter Pod advanced affinity | `{}` | +| `replsets.rs0.arbiter.tolerations` | MongoDB arbiter Pod tolerations | `[]` | +| `replsets.rs0.arbiter.priorityClass` | MongoDB arbiter priorityClassName | `""` | +| `replsets.rs0.arbiter.annotations` | MongoDB arbiter Pod annotations | `{}` | +| `replsets.rs0.arbiter.labels` | MongoDB arbiter Pod labels | `{}` | +| `replsets.rs0.arbiter.nodeSelector` | MongoDB arbiter Pod nodeSelector labels | `{}` | +| | +| `sharding.enabled` | Enable sharding setup | `true` | +| `sharding.balancer.enabled` | Enable/disable balancer | `true` | +| `sharding.configrs.size` | Config ReplicaSet size (pod quantity) | `3` | +| `sharding.configrs.terminationGracePeriodSeconds` | The amount of seconds Kubernetes will wait for a clean replica set Pods termination | `""` | +| `sharding.configrs.externalNodes.host` | The URL or IP address of the external config server instance | `""` | +| `sharding.configrs.externalNodes.port` | The port number of the external config server instance | `""` | +| `sharding.configrs.externalNodes.votes` | The number of [votes :octicons-link-external-16:](https://docs.mongodb.com/manual/reference/replica-configuration/#mongodb-rsconf-rsconf.members-n-.votes) of the [external config server instance](replication-main.md) | `""` | +| `sharding.configrs.externalNodes.priority` | The [priority :octicons-link-external-16:](https://docs.mongodb.com/manual/reference/replica-configuration/#mongodb-rsconf-rsconf.members-n-.priority) of the [external config server instance](replication-main.md) | `""` | +| `sharding.configrs.configuration` | Custom config for mongod in config replica set | `""` | +| `sharding.configrs.topologySpreadConstraints` | Control how Pods are spread across your cluster among failure-domains such as regions, zones, nodes, and other user-defined topology domains | `{}` | +| `sharding.configrs.serviceAccountName` | Run sharding configrs Containers under specified K8S SA | `""` | +| `sharding.configrs.affinity.antiAffinityTopologyKey` | Config ReplicaSet Pod affinity | `kubernetes.io/hostname` | +| `sharding.configrs.affinity.advanced` | Config ReplicaSet Pod advanced affinity | `{}` | +| `sharding.configrs.tolerations` | Config ReplicaSet Pod tolerations | `[]` | +| `sharding.configrs.priorityClass` | Config ReplicaSet Pod priorityClassName | `""` | +| `sharding.configrs.annotations` | Config ReplicaSet Pod annotations | `{}` | +| `sharding.configrs.labels` | Config ReplicaSet Pod labels | `{}` | +| `sharding.configrs.nodeSelector` | Config ReplicaSet Pod nodeSelector labels | `{}` | +| `sharding.configrs.livenessProbe` | Config ReplicaSet Pod livenessProbe structure | `{}` | +| `sharding.configrs.readinessProbe` | Config ReplicaSet Pod readinessProbe structure | `{}` | +| `sharding.configrs.storage` | Set cacheSizeRatio or other custom MongoDB storage options | `{}` | +| `sharding.configrs.podSecurityContext` | Set the security context for a Pod | `{}` | +| `sharding.configrs.containerSecurityContext` | Set the security context for a Container | `{}` | +| `sharding.configrs.runtimeClass` | Config ReplicaSet Pod runtimeClassName | `""` | +| `sharding.configrs.sidecars` | Config ReplicaSet Pod sidecars | `{}` | +| `sharding.configrs.sidecarVolumes` | Config ReplicaSet Pod sidecar volumes | `[]` | +| `sharding.configrs.sidecarPVCs` | Config ReplicaSet Pod sidecar PVCs | `[]` | +| `sharding.configrs.podDisruptionBudget.maxUnavailable` | Config ReplicaSet failed Pods maximum quantity | `1` | +| `sharding.configrs.expose.enabled` | Allow access to cfg replica from outside of Kubernetes | `false` | +| `sharding.configrs.expose.type` | Network service access point type | `ClusterIP` | +| `sharding.configrs.expose.loadBalancerIP` | Set client IP to Load Balancer | `""` | +| `sharding.configrs.expose.loadBalancerSourceRanges` | Limit client IP's access to Load Balancer | `{}` | +| `sharding.configrs.expose.annotations` | Config ReplicaSet service annotations | `{}` | +| `sharding.configrs.expose.labels` | Config ReplicaSet service labels | `{}` | +| `sharding.configrs.expose.internalTrafficPolicy` | Config ReplicaSet service internal traffic policy | `Local` | +| `sharding.configrs.expose.externalTrafficPolicy` | Config ReplicaSet service external traffic policy | `Local` | +| `sharding.configrs.resources.limits.cpu` | Config ReplicaSet resource limits CPU | `300m` | +| `sharding.configrs.resources.limits.memory` | Config ReplicaSet resource limits memory | `0.5G` | +| `sharding.configrs.resources.requests.cpu` | Config ReplicaSet resource requests CPU | `300m` | +| `sharding.configrs.resources.requests.memory` | Config ReplicaSet resource requests memory | `0.5G` | +| `sharding.configrs.volumeSpec.hostPath` | Config ReplicaSet hostPath K8S storage | | +| `sharding.configrs.volumeSpec.hostPath.path` | Config ReplicaSet hostPath K8S storage path | `""` | +| `sharding.configrs.volumeSpec.hostPath.type` | Type for hostPath volum | `Directory` | +| `sharding.configrs.volumeSpec.emptyDir` | Config ReplicaSet Pods emptyDir K8S storage | | +| `sharding.configrs.volumeSpec.pvc` | Config ReplicaSet Pods PVC request parameters | | +| `sharding.configrs.volumeSpec.pvc.annotations` | The Kubernetes annotations metadata for Persistent Volume Claim | `{}` | +| `sharding.configrs.volumeSpec.pvc.labels` | The Kubernetes labels metadata for Persistent Volume Claim | `{}` | +| `sharding.configrs.volumeSpec.pvc.storageClassName` | Config ReplicaSet Pods PVC storageClass | `""` | +| `sharding.configrs.volumeSpec.pvc.accessModes` | Config ReplicaSet Pods PVC access policy | `[]` | +| `sharding.configrs.volumeSpec.pvc.resources.requests.storage` | Config ReplicaSet Pods PVC storage size | `3Gi` | +| `sharding.configrs.hostAliases` | The IP address for Kubernetes host aliases | `[]` | +| `sharding.mongos.size` | Mongos size (pod quantity) | `3` | +| `sharding.mongos.terminationGracePeriodSeconds` | The amount of seconds Kubernetes will wait for a clean mongos Pods termination | `""` | +| `sharding.mongos.configuration` | Custom config for mongos | `""` | +| `sharding.mongos.topologySpreadConstraints` | Control how Pods are spread across your cluster among failure-domains such as regions, zones, nodes, and other user-defined topology domains | `{}` | +| `sharding.mongos.serviceAccountName` | Run sharding mongos Containers under specified K8S SA | `""` | +| `sharding.mongos.affinity.antiAffinityTopologyKey` | Mongos Pods affinity | `kubernetes.io/hostname` | +| `sharding.mongos.affinity.advanced` | Mongos Pods advanced affinity | `{}` | +| `sharding.mongos.tolerations` | Mongos Pods tolerations | `[]` | +| `sharding.mongos.priorityClass` | Mongos Pods priorityClassName | `""` | +| `sharding.mongos.annotations` | Mongos Pods annotations | `{}` | +| `sharding.mongos.labels` | Mongos Pods labels | `{}` | +| `sharding.mongos.nodeSelector` | Mongos Pods nodeSelector labels | `{}` | +| `sharding.mongos.livenessProbe` | Mongos Pod livenessProbe structure | `{}` | +| `sharding.mongos.readinessProbe` | Mongos Pod readinessProbe structure | `{}` | +| `sharding.mongos.podSecurityContext` | Set the security context for a Pod | `{}` | +| `sharding.mongos.containerSecurityContext` | Set the security context for a Container | `{}` | +| `sharding.mongos.runtimeClass` | Mongos Pod runtimeClassName | `""` | +| `sharding.mongos.sidecars` | Mongos Pod sidecars | `{}` | +| `sharding.mongos.sidecarVolumes` | Mongos Pod sidecar volumes | `[]` | +| `sharding.mongos.sidecarPVCs` | Mongos Pod sidecar PVCs | `[]` | +| `sharding.mongos.podDisruptionBudget.maxUnavailable` | Mongos failed Pods maximum quantity | `1` | +| `sharding.mongos.resources.limits.cpu` | Mongos Pods resource limits CPU | `300m` | +| `sharding.mongos.resources.limits.memory` | Mongos Pods resource limits memory | `0.5G` | +| `sharding.mongos.resources.requests.cpu` | Mongos Pods resource requests CPU | `300m` | +| `sharding.mongos.resources.requests.memory` | Mongos Pods resource requests memory | `0.5G` | +| `sharding.mongos.expose.type` | Mongos service type | `ClusterIP` | +| `sharding.mongos.expose.loadBalancerIP` | Set client IP to Load Balancer | `""` | +| `sharding.mongos.expose.servicePerPod` | Create a separate ClusterIP Service for each mongos instance | `false` | +| `sharding.mongos.expose.loadBalancerSourceRanges` | Limit client IP's access to Load Balancer | `{}` | +| `sharding.mongos.expose.annotations` | Mongos service annotations | `{}` | +| `sharding.mongos.expose.labels` | Mongos service labels | `{}` | +| `sharding.mongos.expose.internalTrafficPolicy` | Mongos service internal traffic policy | `Local` | +| `sharding.mongos.expose.externalTrafficPolicy` | Mongos service external traffic policy | `Local` | +| `sharding.mongos.expose.nodePort` | Custom port if exposing mongos via NodePort | `""` | +| `sharding.mongos.hostAliases` | The IP address for Kubernetes host aliases | `[]` | +| | +| `users.name` | The username of the MongoDB application user | `""` | +| `users.db` | Database that the user authenticates against | `""` | +| `users.passwordSecretRef.name` | Name of the secret that contains the user's password | `""` | +| `users.passwordSecretRef.key` | Key in the secret that corresponds to the value of the user's password | `""` | +| `users.roles.role.name` | Name of the MongoDB role assigned to the user. As [built-in roles](https://www.mongodb.com/docs/manual/reference/built-in-roles/#built-in-roles), so [custom roles](https://github.com/mongodb/mongodb-kubernetes-operator/blob/master/docs/deploy-configure.md#define-a-custom-database-role) are supported | `""` | +| `users.roles.role.db` | Database that the MongoDB role applies to | `""` | +| | +| `roles.role` | Name of the custom role. | `""` | +| `roles.db` | Database in which you want to store the user-defined role. | `admin` | +| `roles.authenticationRestrictions.clientSource` | Array of IP addresses or CIDR blocks from which users assigned this role can connect.MongoDB servers reject connection requests from users with this role if the requests come from a client that is not present in this array. | `""` | +| `roles.authenticationRestrictions.serverAddress` | Array of IP addresses or CIDR blocks to which users assigned this role can connect.MongoDB servers reject connection requests from users with this role if the client requests to connect to a server that is not present in this array. | `""` | +| `roles.privileges.actions` | Name of the role. Valid values are built-in roles. | `[]` | +| `roles.privileges.resource.db` | Database for which the privilege `security.roles.privileges.actions` apply. An empty string ("") indicates that the privilege actions apply to all databases. | `""` | +| `roles.privileges.resource.collection` | Collection for which the privilege `security.roles.privileges.actions` apply. An empty string ("") indicates that the privilege actions apply to all of the database's collections. | `""` | +| `roles.privileges.resource.cluster` | Flag that indicates that the privilege `security.roles.privileges.actions` apply to all databases and collections in the MongoDB deployment. If omitted, defaults to false.If set to true, do not provide values for `security.roles.privileges.resource.database` and `security.roles.privileges.resource.collection`. | `""` | +| `roles.roles.role` | Name of the role to inherit from. | `""` | +| `roles.roles.db` | Name of database that contains the role to inherit from. | `""` | +| | +| `backup.enabled` | Enable backup PBM agent | `true` | +| `backup.annotations` | Backup job annotations | `{}` | +| `backup.podSecurityContext` | Set the security context for a Pod | `{}` | +| `backup.containerSecurityContext` | Set the security context for a Container | `{}` | +| `backup.restartOnFailure` | Backup Pods restart policy | `true` | +| `backup.image.repository` | PBM Container image repository | `percona/percona-backup-mongodb` | +| `backup.image.tag` | PBM Container image tag | `2.7.0-multi` | +| `backup.storages` | Local/remote backup storages settings | `{}` | +| `backup.pitr.enabled` | Enable point in time recovery for backup | `false` | +| `backup.pitr.oplogOnly` | Start collecting oplogs even if full logical backup doesn't exist | `false` | +| `backup.pitr.oplogSpanMin` | Number of minutes between the uploads of oplogs | `10` | +| `backup.pitr.compressionType` | The point-in-time-recovery chunks compression format | `""` | +| `backup.pitr.compressionLevel` | The point-in-time-recovery chunks compression level | `""` | +| `backup.configuration.backupOptions` | Custom configuration settings for backup | `{}` | +| `backup.configuration.restoreOptions` | Custom configuration settings for restore | `{}` | +| `backup.tasks` | Backup working schedule | `{}` | +| `systemUsers` | PSMDB operator system users | `{}` | + +Specify parameters using `--set key=value[,key=value]` argument to `helm install` +Notice that you can use multiple replica sets only with sharding enabled. + +## Examples + +### Deploy a replica set with disabled backups and no mongos pods + +This is great for a dev PSMDB/MongoDB cluster as it doesn't bother with backups and sharding setup. + +```bash +$ helm install dev --namespace psmdb . \ + --set runUid=1001 --set "replsets.rs0.volumeSpec.pvc.resources.requests.storage=20Gi" \ + --set backup.enabled=false --set sharding.enabled=false +``` diff --git a/charts/percona/psmdb-db/1.18.0/templates/NOTES.txt b/charts/percona/psmdb-db/1.18.0/templates/NOTES.txt new file mode 100644 index 0000000000..3f294ce22a --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/templates/NOTES.txt @@ -0,0 +1,40 @@ +# + + % _____ + %%% | __ \ + ###%%%%%%%%%%%%* | |__) |__ _ __ ___ ___ _ __ __ _ + ### ##%% %%%% | ___/ _ \ '__/ __/ _ \| '_ \ / _` | + #### ##% %%%% | | | __/ | | (_| (_) | | | | (_| | + ### #### %%% |_| \___|_| \___\___/|_| |_|\__,_| + ,((### ### %%% _ _ _____ _ + (((( (### #### %%%% | | / _ \ / ____| | | + ((( ((# ###### | | _| (_) |___ | (___ __ _ _ _ __ _ __| | + (((( (((# #### | |/ /> _ >> https://percona.com/k8s <<< + +Percona Server for MongoDB cluster is deployed now. Get the username and password: + + ADMIN_USER=$(kubectl -n {{ .Release.Namespace }} get secrets {{ include "psmdb-database.fullname" . }}-secrets -o jsonpath="{.data.MONGODB_USER_ADMIN_USER}" | base64 --decode) + ADMIN_PASSWORD=$(kubectl -n {{ .Release.Namespace }} get secrets {{ include "psmdb-database.fullname" . }}-secrets -o jsonpath="{.data.MONGODB_USER_ADMIN_PASSWORD}" | base64 --decode) + +Connect to the cluster: +{{- if .Values.sharding.enabled }} + + kubectl run -i --rm --tty percona-client --image=percona/percona-server-mongodb:7.0 --restart=Never \ + -- mongosh "mongodb://${ADMIN_USER}:${ADMIN_PASSWORD}@{{ include "psmdb-database.fullname" . }}-mongos.{{ .Release.Namespace }}.svc.cluster.local/admin?ssl=false" + +{{- else }} + + kubectl run -i --rm --tty percona-client --image=percona/percona-server-mongodb:7.0 --restart=Never \ + -- mongosh "mongodb+srv://${ADMIN_USER}:${ADMIN_PASSWORD}@{{ include "psmdb-database.fullname" . }}-{{ .Values.replsets.rs0.name }}.{{ .Release.Namespace }}.svc.cluster.local/admin?replicaSet=rs0&ssl=false" + +{{- end }} + diff --git a/charts/percona/psmdb-db/1.18.0/templates/_helpers.tpl b/charts/percona/psmdb-db/1.18.0/templates/_helpers.tpl new file mode 100644 index 0000000000..47b33a5a76 --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/templates/_helpers.tpl @@ -0,0 +1,45 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "psmdb-database.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If release name contains chart name it will be used as a full name. +*/}} +{{- define "psmdb-database.fullname" -}} +{{- if .Values.fullnameOverride -}} +{{- .Values.fullnameOverride | trunc 21 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 21 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 21 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "psmdb-database.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 21 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Common labels +*/}} +{{- define "psmdb-database.labels" -}} +app.kubernetes.io/name: {{ include "psmdb-database.name" . }} +helm.sh/chart: {{ include "psmdb-database.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} diff --git a/charts/percona/psmdb-db/1.18.0/templates/cluster-secret.yaml b/charts/percona/psmdb-db/1.18.0/templates/cluster-secret.yaml new file mode 100644 index 0000000000..a3bf7a79e5 --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/templates/cluster-secret.yaml @@ -0,0 +1,12 @@ +{{- if hasKey .Values "systemUsers" }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "psmdb-database.fullname" . }}-secrets + namespace: {{ .Release.Namespace }} + labels: +{{ include "psmdb-database.labels" . | indent 4 }} +type: Opaque +stringData: +{{ .Values.systemUsers | toYaml | indent 2 }} +{{- end -}} diff --git a/charts/percona/psmdb-db/1.18.0/templates/cluster.yaml b/charts/percona/psmdb-db/1.18.0/templates/cluster.yaml new file mode 100644 index 0000000000..165bba3d17 --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/templates/cluster.yaml @@ -0,0 +1,676 @@ +apiVersion: psmdb.percona.com/v1 +kind: PerconaServerMongoDB +metadata: + {{- if .Values.annotations }} + annotations: +{{ .Values.annotations | toYaml | indent 4 }} + {{- end }} + name: {{ include "psmdb-database.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "psmdb-database.labels" . | indent 4 }} + finalizers: +{{ .Values.finalizers | toYaml | indent 4 }} +spec: + crVersion: {{ .Values.crVersion }} + pause: {{ .Values.pause }} + unmanaged: {{ .Values.unmanaged }} + {{- if .Values.platform }} + platform: {{ .Values.platform }} + {{- end }} + enableVolumeExpansion: {{ .Values.enableVolumeExpansion }} + {{- if .Values.clusterServiceDNSSuffix }} + clusterServiceDNSSuffix: {{ .Values.clusterServiceDNSSuffix }} + {{- end }} + {{- if .Values.clusterServiceDNSMode }} + clusterServiceDNSMode: {{ .Values.clusterServiceDNSMode }} + {{- end }} + image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" + imagePullPolicy: "{{ .Values.imagePullPolicy }}" + {{- if .Values.unsafeFlags }} + unsafeFlags: +{{ .Values.unsafeFlags | toYaml | indent 4 }} + {{- end }} + {{- if .Values.ignoreAnnotations }} + ignoreAnnotations: +{{ .Values.ignoreAnnotations | toYaml | indent 4 }} + {{- end }} + {{- if .Values.ignoreLabels }} + ignoreLabels: +{{ .Values.ignoreLabels | toYaml | indent 4 }} + {{- end }} + multiCluster: + enabled: {{ .Values.multiCluster.enabled }} + {{- if .Values.multiCluster.DNSSuffix }} + DNSSuffix: {{ .Values.multiCluster.DNSSuffix }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ .Values.imagePullSecrets | toYaml | indent 4 }} + {{- end }} + {{- if .Values.initImage }} + initImage: "{{ .Values.initImage.repository }}:{{ .Values.initImage.tag }}" + {{- end }} + {{- if .Values.initContainerSecurityContext }} + initContainerSecurityContext: +{{ .Values.initContainerSecurityContext | toYaml | indent 4 }} + {{- end }} + {{- if .Values.tls }} + tls: +{{ .Values.tls | toYaml | indent 4 }} + {{- end }} + {{- if .Values.secrets }} + secrets: +{{ tpl (.Values.secrets | toYaml) . | indent 4 }} + {{- else }} + secrets: + users: {{ include "psmdb-database.fullname" . }}-secrets + {{- end }} + {{- if .Values.updateStrategy }} + updateStrategy: {{ .Values.updateStrategy }} + upgradeOptions: + versionServiceEndpoint: {{ .Values.upgradeOptions.versionServiceEndpoint }} + apply: {{ .Values.upgradeOptions.apply }} + schedule: {{ .Values.upgradeOptions.schedule }} + setFCV: {{ .Values.upgradeOptions.setFCV }} + {{- end }} + pmm: + enabled: {{ .Values.pmm.enabled }} + image: "{{ .Values.pmm.image.repository }}:{{ .Values.pmm.image.tag }}" + serverHost: {{ .Values.pmm.serverHost }} + {{- if .Values.pmm.mongodParams }} + mongodParams: {{ .Values.pmm.mongodParams }} + {{- end }} + {{- if .Values.pmm.mongosParams }} + mongosParams: {{ .Values.pmm.mongosParams }} + {{- end }} + {{- if .Values.pmm.resources }} + resources: +{{ .Values.pmm.resources | toYaml | indent 6 }} + {{- end }} + {{- if .Values.pmm.containerSecurityContext }} + containerSecurityContext: +{{ .Values.pmm.containerSecurityContext | toYaml | indent 6 }} + {{- end }} + + replsets: + {{- range $k,$replset := .Values.replsets }} + {{- if $replset.name }} + - name: {{ $replset.name }} + {{- else }} + - name: {{ $k }} + {{- end }} + size: {{ $replset.size }} + {{- if $replset.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ $replset.terminationGracePeriodSeconds }} + {{- end }} + {{- if $replset.externalNodes }} + externalNodes: +{{ $replset.externalNodes | toYaml | indent 6 }} + {{- end }} + {{- if $replset.configuration }} + configuration: | +{{ $replset.configuration | indent 6 }} + {{- end }} + {{- if $replset.topologySpreadConstraints }} + topologySpreadConstraints: +{{ $replset.topologySpreadConstraints | toYaml | indent 6 }} + {{- end }} +{{- if $replset.replsetOverrides }} + replsetOverrides: +{{ $replset.replsetOverrides | toYaml | indent 6 }} + {{- end }} + {{- if $replset.serviceAccountName }} + serviceAccountName: {{ $replset.serviceAccountName }} + {{- end }} + {{- if $replset.affinity }} + affinity: +{{ $replset.affinity | toYaml | indent 6 }} + {{- end }} + {{- if $replset.priorityClass }} + priorityClassName: {{ $replset.priorityClass }} + {{- end }} + {{- if $replset.primaryPreferTagSelector }} + primaryPreferTagSelector: +{{ $replset.primaryPreferTagSelector | toYaml | indent 6 }} + {{- end }} + {{- if $replset.annotations }} + annotations: +{{ $replset.annotations | toYaml | indent 6 }} + {{- end }} + {{- if $replset.labels }} + labels: +{{ $replset.labels | toYaml | indent 6 }} + {{- end }} + {{- if $replset.nodeSelector }} + nodeSelector: +{{ $replset.nodeSelector | toYaml | indent 6 }} + {{- end }} + {{- if $replset.tolerations }} + tolerations: +{{ $replset.tolerations | toYaml | indent 6 }} + {{- end }} + {{- if $replset.livenessProbe }} + livenessProbe: +{{ $replset.livenessProbe | toYaml | indent 6 }} + {{- end }} + {{- if $replset.readinessProbe }} + readinessProbe: +{{ $replset.readinessProbe | toYaml | indent 6 }} + {{- end }} + {{- if $replset.storage }} + storage: +{{ $replset.storage | toYaml | indent 6 }} + {{- end }} + {{- if $replset.podSecurityContext }} + podSecurityContext: +{{ $replset.podSecurityContext | toYaml | indent 6 }} + {{- end }} + {{- if $replset.containerSecurityContext }} + containerSecurityContext: +{{ $replset.containerSecurityContext | toYaml | indent 6 }} + {{- end }} + {{- if $replset.runtimeClass }} + runtimeClassName: {{ $replset.runtimeClass }} + {{- end }} + {{- if $replset.sidecars }} + sidecars: +{{ $replset.sidecars | toYaml | indent 6 }} + {{- end }} + {{- if $replset.sidecarVolumes }} + sidecarVolumes: +{{ $replset.sidecarVolumes | toYaml | indent 6 }} + {{- end }} + {{- if $replset.sidecarPVCs }} + sidecarPVCs: +{{ $replset.sidecarPVCs | toYaml | indent 6 }} + {{- end }} + {{- if $replset.podDisruptionBudget }} + podDisruptionBudget: + {{- if $replset.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ $replset.podDisruptionBudget.maxUnavailable }} + {{- else }} + minAvailable: {{ $replset.podDisruptionBudget.minAvailable }} + {{- end }} + {{- end }} + {{- if $replset.splitHorizons }} + splitHorizons: +{{ $replset.splitHorizons | toYaml | indent 6 }} + {{- end }} + {{- if $replset.expose }} + expose: + enabled: {{ $replset.expose.enabled }} + type: {{ $replset.expose.type }} + {{- if $replset.expose.loadBalancerIP }} + loadBalancerIP: {{ $replset.expose.loadBalancerIP }} + {{- end }} + {{- if $replset.expose.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ $replset.expose.loadBalancerSourceRanges | toYaml | indent 8 }} + {{- end }} + {{- if $replset.expose.annotations }} + annotations: +{{ $replset.expose.annotations | toYaml | indent 8 }} + {{- end }} + {{- if $replset.expose.labels }} + labels: +{{ $replset.expose.labels | toYaml | indent 8 }} + {{- end }} + {{- if $replset.expose.internalTrafficPolicy }} + internalTrafficPolicy: {{ $replset.expose.internalTrafficPolicy }} + {{- end }} + {{- if $replset.expose.externalTrafficPolicy }} + externalTrafficPolicy: {{ $replset.expose.externalTrafficPolicy }} + {{- end }} + {{- end }} + {{- if $replset.schedulerName }} + schedulerName: {{ $replset.schedulerName }} + {{- end }} + resources: + {{- if $replset.resources }} +{{ $replset.resources | toYaml | indent 6 }} + {{- end }} + volumeSpec: + {{- if $replset.volumeSpec }} + {{- if $replset.volumeSpec.hostPath }} + hostPath: + path: {{ $replset.volumeSpec.hostPath.path }} + {{- if $replset.volumeSpec.hostPath.type }} + type: {{ $replset.volumeSpec.hostPath.type }} + {{- else }} + type: Directory + {{- end }} + {{- else if $replset.volumeSpec.pvc }} + persistentVolumeClaim: +{{ $replset.volumeSpec.pvc | toYaml | indent 8 }} + {{- else }} + emptyDir: {} + {{- end }} + {{- end }} + {{- if $replset.hostAliases }} + hostAliases: +{{ $replset.hostAliases | toYaml | indent 6 }} + {{- end }} + {{- if $replset.nonvoting }} + nonvoting: + enabled: {{ $replset.nonvoting.enabled }} + size: {{ $replset.nonvoting.size }} + {{- if $replset.nonvoting.configuration }} + configuration: | +{{ $replset.nonvoting.configuration | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.serviceAccountName }} + serviceAccountName: {{ $replset.nonvoting.serviceAccountName }} + {{- end }} + affinity: + {{- if $replset.nonvoting.affinity }} +{{ $replset.nonvoting.affinity | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.priorityClass }} + priorityClassName: {{ $replset.nonvoting.priorityClass }} + {{- end }} + {{- if $replset.nonvoting.annotations }} + annotations: +{{ $replset.nonvoting.annotations | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.labels }} + labels: +{{ $replset.nonvoting.labels | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.podSecurityContext }} + podSecurityContext: +{{ $replset.nonvoting.podSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.containerSecurityContext }} + containerSecurityContext: +{{ $replset.nonvoting.containerSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.nodeSelector }} + nodeSelector: +{{ $replset.nonvoting.nodeSelector | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.tolerations }} + tolerations: +{{ $replset.nonvoting.tolerations | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.podDisruptionBudget }} + podDisruptionBudget: + {{- if $replset.nonvoting.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ $replset.nonvoting.podDisruptionBudget.maxUnavailable }} + {{- else }} + minAvailable: {{ $replset.nonvoting.podDisruptionBudget.minAvailable }} + {{- end }} + {{- end }} + {{- if $replset.nonvoting.resources }} + resources: +{{ $replset.nonvoting.resources | toYaml | indent 8 }} + {{- end }} + {{- if $replset.nonvoting.volumeSpec }} + volumeSpec: + {{- if $replset.nonvoting.volumeSpec.hostPath }} + hostPath: + path: {{ $replset.nonvoting.volumeSpec.hostPath.path }} + {{- if $replset.nonvoting.volumeSpec.hostPath.type }} + type: {{ $replset.nonvoting.volumeSpec.hostPath.type }} + {{- else }} + type: Directory + {{- end }} + {{- else if $replset.nonvoting.volumeSpec.pvc }} + persistentVolumeClaim: +{{ $replset.nonvoting.volumeSpec.pvc | toYaml | indent 10 }} + {{- else }} + emptyDir: {} + {{- end }} + {{- end }} + {{- end }} + {{- if $replset.arbiter }} + arbiter: + enabled: {{ $replset.arbiter.enabled }} + size: {{ $replset.arbiter.size }} + {{- if $replset.arbiter.serviceAccountName }} + serviceAccountName: {{ $replset.arbiter.serviceAccountName }} + {{- end }} + {{- if $replset.affinity }} + affinity: +{{ $replset.arbiter.affinity | toYaml | indent 8 }} + {{- end }} + {{- if $replset.arbiter.priorityClass }} + priorityClassName: {{ $replset.arbiter.priorityClass }} + {{- end }} + {{- if $replset.arbiter.annotations }} + annotations: +{{ $replset.arbiter.annotations | toYaml | indent 8 }} + {{- end }} + {{- if $replset.arbiter.labels }} + labels: +{{ $replset.arbiter.labels | toYaml | indent 8 }} + {{- end }} + {{- if $replset.arbiter.nodeSelector }} + nodeSelector: +{{ $replset.arbiter.nodeSelector | toYaml | indent 8 }} + {{- end }} + {{- if $replset.arbiter.tolerations }} + tolerations: +{{ $replset.arbiter.tolerations | toYaml | indent 8 }} + {{- end }} + {{- end }} + {{- end }} + + sharding: + enabled: {{ .Values.sharding.enabled }} + balancer: + enabled: {{ .Values.sharding.balancer.enabled }} + + configsvrReplSet: + size: {{ .Values.sharding.configrs.size }} + {{- if .Values.sharding.configrs.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.sharding.configrs.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.sharding.configrs.externalNodes }} + externalNodes: +{{ .Values.sharding.configrs.externalNodes | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.configuration }} + configuration: | +{{ .Values.sharding.configrs.configuration | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.topologySpreadConstraints }} + topologySpreadConstraints: +{{ .Values.sharding.configrs.topologySpreadConstraints | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.serviceAccountName }} + serviceAccountName: {{ .Values.sharding.configrs.serviceAccountName }} + {{- end }} + affinity: +{{ .Values.sharding.configrs.affinity | toYaml | indent 8 }} + {{- if .Values.sharding.configrs.priorityClass }} + priorityClassName: {{ .Values.sharding.configrs.priorityClass }} + {{- end }} + {{- if .Values.sharding.configrs.annotations }} + annotations: +{{ .Values.sharding.configrs.annotations | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.labels }} + labels: +{{ .Values.sharding.configrs.labels | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.nodeSelector }} + nodeSelector: +{{ .Values.sharding.configrs.nodeSelector | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.tolerations }} + tolerations: +{{ .Values.sharding.configrs.tolerations | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.livenessProbe }} + livenessProbe: +{{ .Values.sharding.configrs.livenessProbe | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.readinessProbe }} + readinessProbe: +{{ .Values.sharding.configrs.readinessProbe | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.storage }} + storage: +{{ .Values.sharding.configrs.storage | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.podSecurityContext }} + podSecurityContext: +{{ .Values.sharding.configrs.podSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.containerSecurityContext }} + containerSecurityContext: +{{ .Values.sharding.configrs.containerSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.runtimeClass }} + runtimeClassName: {{ .Values.sharding.configrs.runtimeClass }} + {{- end }} + {{- if .Values.sharding.configrs.sidecars }} + sidecars: +{{ .Values.sharding.configrs.sidecars | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.sidecarVolumes }} + sidecarVolumes: +{{ .Values.sharding.configrs.sidecarVolumes | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.sidecarPVCs }} + sidecarPVCs: +{{ .Values.sharding.configrs.sidecarPVCs | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.configrs.podDisruptionBudget }} + podDisruptionBudget: + {{- if .Values.sharding.configrs.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ .Values.sharding.configrs.podDisruptionBudget.maxUnavailable }} + {{- else }} + minAvailable: {{ .Values.sharding.configrs.podDisruptionBudget.minAvailable }} + {{- end }} + {{- end }} + {{- if .Values.sharding.configrs.expose }} + expose: + enabled: {{ .Values.sharding.configrs.expose.enabled }} + type: {{ .Values.sharding.configrs.expose.type }} + {{- if .Values.sharding.configrs.expose.loadBalancerIP }} + loadBalancerIP: {{ .Values.sharding.configrs.expose.loadBalancerIP }} + {{- end }} + {{- if .Values.sharding.configrs.expose.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ .Values.sharding.configrs.expose.loadBalancerSourceRanges | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.configrs.expose.annotations }} + annotations: +{{ .Values.sharding.configrs.expose.annotations | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.configrs.expose.labels }} + labels: +{{ .Values.sharding.configrs.expose.labels | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.configrs.expose.internalTrafficPolicy }} + internalTrafficPolicy: {{ .Values.sharding.configrs.expose.internalTrafficPolicy }} + {{- end }} + {{- if .Values.sharding.configrs.expose.externalTrafficPolicy }} + externalTrafficPolicy: {{ .Values.sharding.configrs.expose.externalTrafficPolicy }} + {{- end }} + {{- end }} + resources: + limits: + cpu: {{ .Values.sharding.configrs.resources.limits.cpu }} + memory: {{ .Values.sharding.configrs.resources.limits.memory }} + requests: + cpu: {{ .Values.sharding.configrs.resources.requests.cpu }} + memory: {{ .Values.sharding.configrs.resources.requests.memory }} + volumeSpec: + {{- if .Values.sharding.configrs.volumeSpec.hostPath }} + hostPath: + path: {{ .Values.sharding.configrs.volumeSpec.hostPath.path }} + {{- if .Values.sharding.configrs.volumeSpec.hostPath.type }} + type: {{ .Values.sharding.configrs.volumeSpec.hostPath.type }} + {{- else }} + type: Directory + {{- end }} + {{- else if .Values.sharding.configrs.volumeSpec.pvc }} + persistentVolumeClaim: +{{ .Values.sharding.configrs.volumeSpec.pvc | toYaml | indent 10 }} + {{- else }} + emptyDir: {} + {{- end }} + {{- if .Values.sharding.configrs.hostAliases }} + hostAliases: +{{ .Values.sharding.configrs.hostAliases | toYaml | indent 8 }} + {{- end }} + + mongos: + size: {{ .Values.sharding.mongos.size }} + {{- if .Values.sharding.mongos.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.sharding.mongos.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.sharding.mongos.configuration }} + configuration: | +{{ .Values.sharding.mongos.configuration | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.topologySpreadConstraints }} + topologySpreadConstraints: +{{ .Values.sharding.mongos.topologySpreadConstraints | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.serviceAccountName }} + serviceAccountName: {{ .Values.sharding.mongos.serviceAccountName }} + {{- end }} + affinity: +{{ .Values.sharding.mongos.affinity | toYaml | indent 8 }} + {{- if .Values.sharding.mongos.priorityClass }} + priorityClassName: {{ .Values.sharding.mongos.priorityClass }} + {{- end }} + {{- if .Values.sharding.mongos.annotations }} + annotations: +{{ .Values.sharding.mongos.annotations | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.labels }} + labels: +{{ .Values.sharding.mongos.labels | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.nodeSelector }} + nodeSelector: +{{ .Values.sharding.mongos.nodeSelector | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.tolerations }} + tolerations: +{{ .Values.sharding.mongos.tolerations | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.livenessProbe }} + livenessProbe: +{{ .Values.sharding.mongos.livenessProbe | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.readinessProbe }} + readinessProbe: +{{ .Values.sharding.mongos.readinessProbe | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.podSecurityContext }} + podSecurityContext: +{{ .Values.sharding.mongos.podSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.containerSecurityContext }} + containerSecurityContext: +{{ .Values.sharding.mongos.containerSecurityContext | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.runtimeClass }} + runtimeClassName: {{ .Values.sharding.mongos.runtimeClass }} + {{- end }} + {{- if .Values.sharding.mongos.sidecars }} + sidecars: +{{ .Values.sharding.mongos.sidecars | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.sidecarVolumes }} + sidecarVolumes: +{{ .Values.sharding.mongos.sidecarVolumes | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.sidecarPVCs }} + sidecarPVCs: +{{ .Values.sharding.mongos.sidecarPVCs | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.podDisruptionBudget }} + podDisruptionBudget: + {{- if .Values.sharding.mongos.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ .Values.sharding.mongos.podDisruptionBudget.maxUnavailable }} + {{- else }} + minAvailable: {{ .Values.sharding.mongos.podDisruptionBudget.minAvailable }} + {{- end }} + {{- end }} + resources: + limits: + cpu: {{ .Values.sharding.mongos.resources.limits.cpu }} + memory: {{ .Values.sharding.mongos.resources.limits.memory }} + requests: + cpu: {{ .Values.sharding.mongos.resources.requests.cpu }} + memory: {{ .Values.sharding.mongos.resources.requests.memory }} + expose: + type: {{ .Values.sharding.mongos.expose.type }} + {{- if .Values.sharding.mongos.expose.loadBalancerIP }} + loadBalancerIP: {{ .Values.sharding.mongos.expose.loadBalancerIP }} + {{- end }} + {{- if .Values.sharding.mongos.expose.servicePerPod }} + servicePerPod: {{ .Values.sharding.mongos.expose.servicePerPod }} + {{- end }} + {{- if .Values.sharding.mongos.expose.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ .Values.sharding.mongos.expose.loadBalancerSourceRanges | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.mongos.expose.annotations }} + annotations: +{{ .Values.sharding.mongos.expose.annotations | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.mongos.expose.labels }} + labels: +{{ .Values.sharding.mongos.expose.labels | toYaml | indent 10 }} + {{- end }} + {{- if .Values.sharding.mongos.expose.internalTrafficPolicy }} + internalTrafficPolicy: {{ .Values.sharding.mongos.expose.internalTrafficPolicy }} + {{- end }} + {{- if .Values.sharding.mongos.expose.externalTrafficPolicy }} + externalTrafficPolicy: {{ .Values.sharding.mongos.expose.externalTrafficPolicy }} + {{- end }} + {{- if .Values.sharding.mongos.expose.nodePort }} + nodePort: {{ .Values.sharding.mongos.expose.nodePort }} + {{- end }} + {{- if .Values.sharding.mongos.auditLog }} + auditLog: +{{ .Values.sharding.mongos.auditLog | toYaml | indent 8 }} + {{- end }} + {{- if .Values.sharding.mongos.hostAliases }} + hostAliases: +{{ .Values.sharding.mongos.hostAliases | toYaml | indent 8 }} + {{- end }} + + {{- if .Values.users }} + users: +{{ .Values.users | toYaml | indent 2 }} + {{- end }} + + {{- if .Values.roles }} + roles: +{{ .Values.roles | toYaml | indent 2 }} + {{- end }} + + backup: + enabled: {{ .Values.backup.enabled }} + {{- if .Values.backup.annotations }} + annotations: +{{ .Values.backup.annotations | toYaml | indent 6 }} + {{- end }} + {{- if .Values.backup.podSecurityContext }} + podSecurityContext: +{{ .Values.backup.podSecurityContext | toYaml | indent 6 }} + {{- end }} + {{- if .Values.backup.containerSecurityContext }} + containerSecurityContext: +{{ .Values.backup.containerSecurityContext | toYaml | indent 6 }} + {{- end }} + image: "{{ .Values.backup.image.repository }}:{{ .Values.backup.image.tag }}" + {{- if .Values.backup.resources }} + resources: +{{ .Values.backup.resources | toYaml | indent 6 }} + {{- end }} + storages: +{{ .Values.backup.storages | toYaml | indent 6 }} + pitr: + {{- if and .Values.backup.enabled .Values.backup.pitr.enabled }} + enabled: true + {{- if .Values.backup.pitr.oplogOnly }} + oplogOnly: {{ .Values.backup.pitr.oplogOnly }} + {{- end }} + {{- if .Values.backup.pitr.oplogSpanMin }} + oplogSpanMin: {{ .Values.backup.pitr.oplogSpanMin }} + {{- end }} + {{- if .Values.backup.pitr.compressionType }} + compressionType: {{ .Values.backup.pitr.compressionType }} + {{- end }} + {{- if .Values.backup.pitr.compressionLevel }} + compressionLevel: {{ .Values.backup.pitr.compressionLevel }} + {{- end }} + {{- else }} + enabled: false + {{- end }} + {{- if .Values.backup.configuration }} + configuration: +{{ .Values.backup.configuration | toYaml | indent 6 }} + {{- end }} + tasks: +{{ .Values.backup.tasks | toYaml | indent 6 }} diff --git a/charts/percona/psmdb-db/1.18.0/values.yaml b/charts/percona/psmdb-db/1.18.0/values.yaml new file mode 100644 index 0000000000..ed1f18d0cf --- /dev/null +++ b/charts/percona/psmdb-db/1.18.0/values.yaml @@ -0,0 +1,678 @@ +# Default values for psmdb-cluster. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# Platform type: kubernetes, openshift +# platform: kubernetes + +# Cluster DNS Suffix +# clusterServiceDNSSuffix: svc.cluster.local +# clusterServiceDNSMode: "Internal" + +finalizers: +## Set this if you want that operator deletes the primary pod last + - percona.com/delete-psmdb-pods-in-order +## Set this if you want to delete database persistent volumes on cluster deletion +# - percona.com/delete-psmdb-pvc +## Set this if you want to delete all pitr chunks on cluster deletion +# - percona.com/delete-pitr-chunks + +nameOverride: "" +fullnameOverride: "" + +crVersion: 1.18.0 +pause: false +unmanaged: false +unsafeFlags: + tls: false + replsetSize: false + mongosSize: false + terminationGracePeriod: false + backupIfUnhealthy: false + +enableVolumeExpansion: false + +annotations: {} + +# ignoreAnnotations: +# - service.beta.kubernetes.io/aws-load-balancer-backend-protocol +# ignoreLabels: +# - rack +multiCluster: + enabled: false + # DNSSuffix: svc.clusterset.local +updateStrategy: SmartUpdate +upgradeOptions: + versionServiceEndpoint: https://check.percona.com + apply: disabled + schedule: "0 2 * * *" + setFCV: false + +image: + repository: percona/percona-server-mongodb + tag: 7.0.14-8-multi + +imagePullPolicy: Always +# imagePullSecrets: [] +# initImage: +# repository: percona/percona-server-mongodb-operator +# tag: 1.18.0 +# initContainerSecurityContext: {} +# tls: +# mode: preferTLS +# # 90 days in hours +# certValidityDuration: 2160h +# allowInvalidCertificates: true +# issuerConf: +# name: special-selfsigned-issuer +# kind: ClusterIssuer +# group: cert-manager.io +secrets: {} + # If you set users secret here the operator will use existing one or generate random values + # If not set the operator generates the default secret with name -secrets + # users: my-cluster-name-secrets + # encryptionKey: my-cluster-name-mongodb-encryption-key + # keyFile: my-cluster-name-mongodb-keyfile + # vault: my-cluster-name-vault + # ldapSecret: my-ldap-secret + # sse: my-cluster-name-sse + +pmm: + enabled: false + image: + repository: percona/pmm-client + tag: 2.43.2 + serverHost: monitoring-service +# mongodParams: "" +# mongosParams: "" +# resources: {} +# containerSecurityContext: {} + +replsets: + rs0: + name: rs0 + size: 3 + # terminationGracePeriodSeconds: 300 + # externalNodes: + # - host: 34.124.76.90 + # - host: 34.124.76.91 + # port: 27017 + # votes: 0 + # priority: 0 + # - host: 34.124.76.92 + # configuration: | + # operationProfiling: + # mode: slowOp + # systemLog: + # verbosity: 1 + # serviceAccountName: percona-server-mongodb-operator + # topologySpreadConstraints: + # - labelSelector: + # matchLabels: + # app.kubernetes.io/name: percona-server-mongodb + # maxSkew: 1 + # topologyKey: kubernetes.io/hostname + # whenUnsatisfiable: DoNotSchedule + # replsetOverrides: + # my-cluster-name-rs0-0: + # host: my-cluster-name-rs0-0.example.net:27017 + # tags: + # key: value-0 + # my-cluster-name-rs0-1: + # host: my-cluster-name-rs0-1.example.net:27017 + # tags: + # key: value-1 + # my-cluster-name-rs0-2: + # host: my-cluster-name-rs0-2.example.net:27017 + # tags: + # key: value-2 + affinity: + antiAffinityTopologyKey: "kubernetes.io/hostname" + # advanced: + # podAffinity: + # requiredDuringSchedulingIgnoredDuringExecution: + # - labelSelector: + # matchExpressions: + # - key: security + # operator: In + # values: + # - S1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # tolerations: [] + # primaryPreferTagSelector: + # region: us-west-2 + # zone: us-west-2c + # priorityClass: "" + # annotations: {} + # labels: {} + # podSecurityContext: {} + # containerSecurityContext: {} + # nodeSelector: {} + # livenessProbe: + # failureThreshold: 4 + # initialDelaySeconds: 60 + # periodSeconds: 30 + # timeoutSeconds: 10 + # startupDelaySeconds: 7200 + # readinessProbe: + # failureThreshold: 8 + # initialDelaySeconds: 10 + # periodSeconds: 3 + # successThreshold: 1 + # timeoutSeconds: 2 + # runtimeClassName: image-rc + # storage: + # engine: wiredTiger + # wiredTiger: + # engineConfig: + # cacheSizeRatio: 0.5 + # directoryForIndexes: false + # journalCompressor: snappy + # collectionConfig: + # blockCompressor: snappy + # indexConfig: + # prefixCompression: true + # inMemory: + # engineConfig: + # inMemorySizeRatio: 0.5 + # sidecars: + # - image: busybox + # command: ["/bin/sh"] + # args: ["-c", "while true; do echo echo $(date -u) 'test' >> /dev/null; sleep 5;done"] + # name: rs-sidecar-1 + # volumeMounts: + # - mountPath: /volume1 + # name: sidecar-volume-claim + # - mountPath: /secret + # name: sidecar-secret + # - mountPath: /configmap + # name: sidecar-config + # sidecarVolumes: + # - name: sidecar-secret + # secret: + # secretName: mysecret + # - name: sidecar-config + # configMap: + # name: myconfigmap + # sidecarPVCs: + # - apiVersion: v1 + # kind: PersistentVolumeClaim + # metadata: + # name: sidecar-volume-claim + # spec: + # resources: + # requests: + # storage: 1Gi + # volumeMode: Filesystem + # accessModes: + # - ReadWriteOnce + podDisruptionBudget: + maxUnavailable: 1 + # splitHorizons: + # my-cluster-name-rs0-0: + # external: rs0-0.mycluster.xyz + # external-2: rs0-0.mycluster2.xyz + # my-cluster-name-rs0-1: + # external: rs0-1.mycluster.xyz + # external-2: rs0-1.mycluster2.xyz + # my-cluster-name-rs0-2: + # external: rs0-2.mycluster.xyz + # external-2: rs0-2.mycluster2.xyz + expose: + enabled: false + type: ClusterIP + # loadBalancerIP: 10.0.0.0 + # loadBalancerSourceRanges: + # - 10.0.0.0/8 + # annotations: + # service.beta.kubernetes.io/aws-load-balancer-backend-protocol: http + # labels: + # some-label: some-key + # internalTrafficPolicy: Local + # schedulerName: "" + resources: + limits: + cpu: "300m" + memory: "0.5G" + requests: + cpu: "300m" + memory: "0.5G" + volumeSpec: + # emptyDir: {} + # hostPath: + # path: /data + # type: Directory + pvc: + # annotations: + # volume.beta.kubernetes.io/storage-class: example-hostpath + # labels: + # rack: rack-22 + # storageClassName: standard + # accessModes: [ "ReadWriteOnce" ] + resources: + requests: + storage: 3Gi + # hostAliases: + # - ip: "10.10.0.2" + # hostnames: + # - "host1" + # - "host2" + nonvoting: + enabled: false + # podSecurityContext: {} + # containerSecurityContext: {} + size: 3 + # configuration: | + # operationProfiling: + # mode: slowOp + # systemLog: + # verbosity: 1 + # serviceAccountName: percona-server-mongodb-operator + affinity: + antiAffinityTopologyKey: "kubernetes.io/hostname" + # advanced: + # podAffinity: + # requiredDuringSchedulingIgnoredDuringExecution: + # - labelSelector: + # matchExpressions: + # - key: security + # operator: In + # values: + # - S1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # tolerations: [] + # priorityClass: "" + # annotations: {} + # labels: {} + # nodeSelector: {} + podDisruptionBudget: + maxUnavailable: 1 + resources: + limits: + cpu: "300m" + memory: "0.5G" + requests: + cpu: "300m" + memory: "0.5G" + volumeSpec: + # emptyDir: {} + # hostPath: + # path: /data + # type: Directory + pvc: + # annotations: + # volume.beta.kubernetes.io/storage-class: example-hostpath + # labels: + # rack: rack-22 + # storageClassName: standard + # accessModes: [ "ReadWriteOnce" ] + resources: + requests: + storage: 3Gi + arbiter: + enabled: false + size: 1 + # serviceAccountName: percona-server-mongodb-operator + affinity: + antiAffinityTopologyKey: "kubernetes.io/hostname" + # advanced: + # podAffinity: + # requiredDuringSchedulingIgnoredDuringExecution: + # - labelSelector: + # matchExpressions: + # - key: security + # operator: In + # values: + # - S1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # tolerations: [] + # priorityClass: "" + # annotations: {} + # labels: {} + # nodeSelector: {} + +sharding: + enabled: true + balancer: + enabled: true + + configrs: + size: 3 + # terminationGracePeriodSeconds: 300 + # externalNodes: + # - host: 34.124.76.90 + # - host: 34.124.76.91 + # port: 27017 + # votes: 0 + # priority: 0 + # - host: 34.124.76.92 + # configuration: | + # operationProfiling: + # mode: slowOp + # systemLog: + # verbosity: 1 + # serviceAccountName: percona-server-mongodb-operator + # topologySpreadConstraints: + # - labelSelector: + # matchLabels: + # app.kubernetes.io/name: percona-server-mongodb + # maxSkew: 1 + # topologyKey: kubernetes.io/hostname + # whenUnsatisfiable: DoNotSchedule + affinity: + antiAffinityTopologyKey: "kubernetes.io/hostname" + # advanced: + # podAffinity: + # requiredDuringSchedulingIgnoredDuringExecution: + # - labelSelector: + # matchExpressions: + # - key: security + # operator: In + # values: + # - S1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # tolerations: [] + # priorityClass: "" + # annotations: {} + # labels: {} + # podSecurityContext: {} + # containerSecurityContext: {} + # nodeSelector: {} + # livenessProbe: {} + # readinessProbe: {} + # runtimeClassName: image-rc + # sidecars: + # - image: busybox + # command: ["/bin/sh"] + # args: ["-c", "while true; do echo echo $(date -u) 'test' >> /dev/null; sleep 5;done"] + # name: rs-sidecar-1 + # volumeMounts: + # - mountPath: /volume1 + # name: sidecar-volume-claim + # sidecarPVCs: [] + # sidecarVolumes: [] + podDisruptionBudget: + maxUnavailable: 1 + expose: + enabled: false + type: ClusterIP + # loadBalancerIP: 10.0.0.0 + # loadBalancerSourceRanges: + # - 10.0.0.0/8 + # annotations: + # service.beta.kubernetes.io/aws-load-balancer-backend-protocol: http + # labels: + # some-label: some-key + # internalTrafficPolicy: Local + resources: + limits: + cpu: "300m" + memory: "0.5G" + requests: + cpu: "300m" + memory: "0.5G" + volumeSpec: + # emptyDir: {} + # hostPath: + # path: /data + # type: Directory + pvc: + # annotations: + # volume.beta.kubernetes.io/storage-class: example-hostpath + # labels: + # rack: rack-22 + # storageClassName: standard + # accessModes: [ "ReadWriteOnce" ] + resources: + requests: + storage: 3Gi + # hostAliases: + # - ip: "10.10.0.2" + # hostnames: + # - "host1" + # - "host2" + + mongos: + size: 3 + # terminationGracePeriodSeconds: 300 + # configuration: | + # systemLog: + # verbosity: 1 + # serviceAccountName: percona-server-mongodb-operator + # topologySpreadConstraints: + # - labelSelector: + # matchLabels: + # app.kubernetes.io/name: percona-server-mongodb + # maxSkew: 1 + # topologyKey: kubernetes.io/hostname + # whenUnsatisfiable: DoNotSchedule + affinity: + antiAffinityTopologyKey: "kubernetes.io/hostname" + # advanced: + # podAffinity: + # requiredDuringSchedulingIgnoredDuringExecution: + # - labelSelector: + # matchExpressions: + # - key: security + # operator: In + # values: + # - S1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # tolerations: [] + # priorityClass: "" + # annotations: {} + # labels: {} + # podSecurityContext: {} + # containerSecurityContext: {} + # nodeSelector: {} + # livenessProbe: {} + # readinessProbe: {} + # runtimeClassName: image-rc + # sidecars: + # - image: busybox + # command: ["/bin/sh"] + # args: ["-c", "while true; do echo echo $(date -u) 'test' >> /dev/null; sleep 5;done"] + # name: rs-sidecar-1 + # volumeMounts: + # - mountPath: /volume1 + # name: sidecar-volume-claim + # sidecarPVCs: [] + # sidecarVolumes: [] + podDisruptionBudget: + maxUnavailable: 1 + resources: + limits: + cpu: "300m" + memory: "0.5G" + requests: + cpu: "300m" + memory: "0.5G" + expose: + enabled: false + type: ClusterIP + # loadBalancerIP: 10.0.0.0/8 + # loadBalancerSourceRanges: + # - 10.0.0.0/8 + # annotations: + # service.beta.kubernetes.io/aws-load-balancer-backend-protocol: http + # labels: + # some-label: some-key + # internalTrafficPolicy: Local + # nodePort: 32017 + # auditLog: + # destination: file + # format: BSON + # filter: '{}' + # hostAliases: + # - ip: "10.10.0.2" + # hostnames: + # - "host1" + # - "host2" + +# users: +# - name: my-user +# db: admin +# passwordSecretRef: +# name: my-user-password +# key: my-user-password-key +# roles: +# - name: clusterAdmin +# db: admin +# - name: userAdminAnyDatabase +# db: admin +# - name: my-usr +# db: admin +# passwordSecretRef: +# name: my-user-pwd +# key: my-user-pwd-key +# roles: +# - name: dbOwner +# db: sometest + +# roles: +# - role: myClusterwideAdmin +# db: admin +# privileges: +# - resource: +# cluster: true +# actions: +# - addShard +# - resource: +# db: config +# collection: '' +# actions: +# - find +# - update +# - insert +# - remove +# roles: +# - role: read +# db: admin +# - role: my-role +# db: myDb +# privileges: +# - resource: +# db: '' +# collection: '' +# actions: +# - find +# authenticationRestrictions: +# - clientSource: +# - 127.0.0.1 +# serverAddress: +# - 127.0.0.1 + + +backup: + enabled: true + image: + repository: percona/percona-backup-mongodb + tag: 2.7.0-multi + # annotations: + # iam.amazonaws.com/role: role-arn + # podSecurityContext: {} + # containerSecurityContext: {} + # resources: + # limits: + # cpu: "300m" + # memory: "1.2G" + # requests: + # cpu: "300m" + # memory: "1G" + storages: + # s3-us-west: + # type: s3 + # s3: + # bucket: S3-BACKUP-BUCKET-NAME-HERE + # credentialsSecret: my-cluster-name-backup-s3 + # serverSideEncryption: + # kmsKeyID: 1234abcd-12ab-34cd-56ef-1234567890ab + # sseAlgorithm: aws:kms + # sseCustomerAlgorithm: AES256 + # sseCustomerKey: Y3VzdG9tZXIta2V5 + # retryer: + # numMaxRetries: 3 + # minRetryDelay: 30ms + # maxRetryDelay: 5m + # region: us-west-2 + # prefix: "" + # uploadPartSize: 10485760 + # maxUploadParts: 10000 + # storageClass: STANDARD + # insecureSkipTLSVerify: false + # minio: + # type: s3 + # s3: + # bucket: MINIO-BACKUP-BUCKET-NAME-HERE + # region: us-east-1 + # credentialsSecret: my-cluster-name-backup-minio + # endpointUrl: http://minio.psmdb.svc.cluster.local:9000/minio/ + # prefix: "" + # azure-blob: + # type: azure + # azure: + # container: CONTAINER-NAME + # prefix: PREFIX-NAME + # endpointUrl: https://accountName.blob.core.windows.net + # credentialsSecret: SECRET-NAME + pitr: + enabled: false + oplogOnly: false + # oplogSpanMin: 10 + # compressionType: gzip + # compressionLevel: 6 + # configuration: + # backupOptions: + # priority: + # "localhost:28019": 2.5 + # "localhost:27018": 2.5 + # timeouts: + # startingStatus: 33 + # oplogSpanMin: 10 + # restoreOptions: + # batchSize: 500 + # numInsertionWorkers: 10 + # numDownloadWorkers: 4 + # maxDownloadBufferMb: 0 + # downloadChunkMb: 32 + # mongodLocation: /usr/bin/mongo + # mongodLocationMap: + # "node01:2017": /usr/bin/mongo + # "node03:27017": /usr/bin/mongo + tasks: + # - name: daily-s3-us-west + # enabled: true + # schedule: "0 0 * * *" + # keep: 3 + # storageName: s3-us-west + # compressionType: gzip + # - name: weekly-s3-us-west + # enabled: false + # schedule: "0 0 * * 0" + # keep: 5 + # storageName: s3-us-west + # compressionType: gzip + # - name: weekly-s3-us-west-physical + # enabled: false + # schedule: "0 5 * * 0" + # keep: 5 + # type: physical + # storageName: s3-us-west + # compressionType: gzip + # compressionLevel: 6 + +# If you set systemUsers here the secret will be constructed by helm with these values +# systemUsers: +# MONGODB_BACKUP_USER: backup +# MONGODB_BACKUP_PASSWORD: backup123456 +# MONGODB_DATABASE_ADMIN_USER: databaseAdmin +# MONGODB_DATABASE_ADMIN_PASSWORD: databaseAdmin123456 +# MONGODB_CLUSTER_ADMIN_USER: clusterAdmin +# MONGODB_CLUSTER_ADMIN_PASSWORD: clusterAdmin123456 +# MONGODB_CLUSTER_MONITOR_USER: clusterMonitor +# MONGODB_CLUSTER_MONITOR_PASSWORD: clusterMonitor123456 +# MONGODB_USER_ADMIN_USER: userAdmin +# MONGODB_USER_ADMIN_PASSWORD: userAdmin123456 +# PMM_SERVER_API_KEY: apikey +# # PMM_SERVER_USER: admin +# # PMM_SERVER_PASSWORD: admin diff --git a/charts/percona/psmdb-operator/1.18.0/.helmignore b/charts/percona/psmdb-operator/1.18.0/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/percona/psmdb-operator/1.18.0/Chart.yaml b/charts/percona/psmdb-operator/1.18.0/Chart.yaml new file mode 100644 index 0000000000..1448b255ed --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/Chart.yaml @@ -0,0 +1,20 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Percona Operator for MongoDB + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: psmdb-operator +apiVersion: v2 +appVersion: 1.18.0 +description: A Helm chart for deploying the Percona Operator for MongoDB +home: https://docs.percona.com/percona-operator-for-mongodb/ +icon: file://assets/icons/psmdb-operator.png +kubeVersion: '>=1.21-0' +maintainers: +- email: tomislav.plavcic@percona.com + name: tplavcic +- email: natalia.marukovich@percona.com + name: nmarukovich +- email: sergey.pronin@percona.com + name: spron-in +name: psmdb-operator +version: 1.18.0 diff --git a/charts/percona/psmdb-operator/1.18.0/LICENSE.txt b/charts/percona/psmdb-operator/1.18.0/LICENSE.txt new file mode 100644 index 0000000000..6a31453af0 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/LICENSE.txt @@ -0,0 +1,13 @@ +Copyright 2019 Paul Czarkowski + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. \ No newline at end of file diff --git a/charts/percona/psmdb-operator/1.18.0/README.md b/charts/percona/psmdb-operator/1.18.0/README.md new file mode 100644 index 0000000000..1cd9b88101 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/README.md @@ -0,0 +1,69 @@ +# Percona Operator for MongoDB + +Percona Operator for MongoDB allows users to deploy and manage Percona Server for MongoDB Clusters on Kubernetes. +Useful links: +- [Operator Github repository](https://github.com/percona/percona-server-mongodb-operator) +- [Operator Documentation](https://www.percona.com/doc/kubernetes-operator-for-psmongodb/index.html) + +## Pre-requisites +* Kubernetes 1.28+ +* Helm v3 + +# Installation + +This chart will deploy the Operator Pod for the further Percona Server for MongoDB creation in Kubernetes. + +## Installing the chart + +To install the chart with the `psmdb` release name using a dedicated namespace (recommended): + +```sh +helm repo add percona https://percona.github.io/percona-helm-charts/ +helm install my-operator percona/psmdb-operator --version 1.18.0 --namespace my-namespace +``` + +The chart can be customized using the following configurable parameters: + +| Parameter | Description | Default | +| ---------------------------- | --------------------------------------------------------------------------------------------------- | ----------------------------------------- | +| `image.repository` | PSMDB Operator Container image name | `percona/percona-server-mongodb-operator` | +| `image.tag` | PSMDB Operator Container image tag | `1.18.0` | +| `image.pullPolicy` | PSMDB Operator Container pull policy | `Always` | +| `image.pullSecrets` | PSMDB Operator Pod pull secret | `[]` | +| `replicaCount` | PSMDB Operator Pod quantity | `1` | +| `tolerations` | List of node taints to tolerate | `[]` | +| `annotations` | PSMDB Operator Deployment annotations | `{}` | +| `podAnnotations` | PSMDB Operator Pod annotations | `{}` | +| `labels` | PSMDB Operator Deployment labels | `{}` | +| `podLabels` | PSMDB Operator Pod labels | `{}` | +| `resources` | Resource requests and limits | `{}` | +| `nodeSelector` | Labels for Pod assignment | `{}` | +| `podAnnotations` | Annotations for pod | `{}` | +| `podSecurityContext` | Pod Security Context | `{}` | +| `watchNamespace` | Set when a different from default namespace is needed to watch (comma separated if multiple needed) | `""` | +| `createNamespace` | Set if you want to create watched namespaces with helm | `false` | +| `rbac.create` | If false RBAC will not be created. RBAC resources will need to be created manually | `true` | +| `securityContext` | Container Security Context | `{}` | +| `serviceAccount.create` | If false the ServiceAccounts will not be created. The ServiceAccounts must be created manually | `true` | +| `serviceAccount.annotations` | PSMDB Operator ServiceAccount annotations | `{}` | +| `logStructured` | Force PSMDB operator to print JSON-wrapped log messages | `false` | +| `logLevel` | PSMDB Operator logging level | `INFO` | +| `disableTelemetry` | Disable sending PSMDB Operator telemetry data to Percona | `false` | + +Specify parameters using `--set key=value[,key=value]` argument to `helm install` + +Alternatively a YAML file that specifies the values for the parameters can be provided like this: + +```sh +helm install psmdb-operator -f values.yaml percona/psmdb-operator +``` + +## Deploy the database + +To deploy Percona Server for MongoDB run the following command: + +```sh +helm install my-db percona/psmdb-db +``` + +See more about Percona Server for MongoDB deployment in its chart [here](https://github.com/percona/percona-helm-charts/tree/main/charts/psmdb-db) or in the [Helm chart installation guide](https://www.percona.com/doc/kubernetes-operator-for-psmongodb/helm.html). diff --git a/charts/percona/psmdb-operator/1.18.0/crds/crd.yaml b/charts/percona/psmdb-operator/1.18.0/crds/crd.yaml new file mode 100644 index 0000000000..89641ef5a6 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/crds/crd.yaml @@ -0,0 +1,19526 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: perconaservermongodbbackups.psmdb.percona.com +spec: + group: psmdb.percona.com + names: + kind: PerconaServerMongoDBBackup + listKind: PerconaServerMongoDBBackupList + plural: perconaservermongodbbackups + shortNames: + - psmdb-backup + singular: perconaservermongodbbackup + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Cluster name + jsonPath: .spec.clusterName + name: Cluster + type: string + - description: Storage name + jsonPath: .spec.storageName + name: Storage + type: string + - description: Backup destination + jsonPath: .status.destination + name: Destination + type: string + - description: Backup type + jsonPath: .status.type + name: Type + type: string + - description: Job status + jsonPath: .status.state + name: Status + type: string + - description: Completed time + jsonPath: .status.completed + name: Completed + type: date + - description: Created time + jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + clusterName: + type: string + compressionLevel: + type: integer + compressionType: + type: string + psmdbCluster: + type: string + storageName: + type: string + type: + enum: + - logical + - physical + type: string + type: object + status: + properties: + azure: + properties: + container: + type: string + credentialsSecret: + type: string + endpointUrl: + type: string + prefix: + type: string + required: + - credentialsSecret + type: object + completed: + format: date-time + type: string + destination: + type: string + error: + type: string + lastTransition: + format: date-time + type: string + latestRestorableTime: + format: date-time + type: string + pbmName: + type: string + pbmPod: + type: string + pbmPods: + additionalProperties: + type: string + type: object + replsetNames: + items: + type: string + type: array + s3: + properties: + bucket: + type: string + credentialsSecret: + type: string + debugLogLevels: + type: string + endpointUrl: + type: string + forcePathStyle: + type: boolean + insecureSkipTLSVerify: + type: boolean + maxUploadParts: + type: integer + prefix: + type: string + region: + type: string + retryer: + properties: + maxRetryDelay: + type: string + minRetryDelay: + type: string + numMaxRetries: + type: integer + type: object + serverSideEncryption: + properties: + kmsKeyID: + type: string + sseAlgorithm: + type: string + sseCustomerAlgorithm: + type: string + sseCustomerKey: + type: string + type: object + storageClass: + type: string + uploadPartSize: + type: integer + required: + - bucket + type: object + start: + format: date-time + type: string + state: + type: string + storageName: + type: string + type: + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: perconaservermongodbrestores.psmdb.percona.com +spec: + group: psmdb.percona.com + names: + kind: PerconaServerMongoDBRestore + listKind: PerconaServerMongoDBRestoreList + plural: perconaservermongodbrestores + shortNames: + - psmdb-restore + singular: perconaservermongodbrestore + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Cluster name + jsonPath: .spec.clusterName + name: Cluster + type: string + - description: Job status + jsonPath: .status.state + name: Status + type: string + - description: Created time + jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + backupName: + type: string + backupSource: + properties: + azure: + properties: + container: + type: string + credentialsSecret: + type: string + endpointUrl: + type: string + prefix: + type: string + required: + - credentialsSecret + type: object + completed: + format: date-time + type: string + destination: + type: string + error: + type: string + lastTransition: + format: date-time + type: string + latestRestorableTime: + format: date-time + type: string + pbmName: + type: string + pbmPod: + type: string + pbmPods: + additionalProperties: + type: string + type: object + replsetNames: + items: + type: string + type: array + s3: + properties: + bucket: + type: string + credentialsSecret: + type: string + debugLogLevels: + type: string + endpointUrl: + type: string + forcePathStyle: + type: boolean + insecureSkipTLSVerify: + type: boolean + maxUploadParts: + type: integer + prefix: + type: string + region: + type: string + retryer: + properties: + maxRetryDelay: + type: string + minRetryDelay: + type: string + numMaxRetries: + type: integer + type: object + serverSideEncryption: + properties: + kmsKeyID: + type: string + sseAlgorithm: + type: string + sseCustomerAlgorithm: + type: string + sseCustomerKey: + type: string + type: object + storageClass: + type: string + uploadPartSize: + type: integer + required: + - bucket + type: object + start: + format: date-time + type: string + state: + type: string + storageName: + type: string + type: + type: string + type: object + clusterName: + type: string + pitr: + properties: + date: + type: string + type: + type: string + type: object + replset: + type: string + selective: + properties: + namespaces: + items: + type: string + type: array + withUsersAndRoles: + type: boolean + type: object + storageName: + type: string + type: object + status: + properties: + completed: + format: date-time + type: string + error: + type: string + lastTransition: + format: date-time + type: string + pbmName: + type: string + pitrTarget: + type: string + state: + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: perconaservermongodbs.psmdb.percona.com +spec: + group: psmdb.percona.com + names: + kind: PerconaServerMongoDB + listKind: PerconaServerMongoDBList + plural: perconaservermongodbs + shortNames: + - psmdb + singular: perconaservermongodb + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-2-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-2-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-3-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-3-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-4-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-4-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-5-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-5-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-6-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-6-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-7-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-7-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-8-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-8-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-9-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-9-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: false + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-10-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-10-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-11-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-11-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + deprecated: true + deprecationWarning: psmdb.percona.com/v1-12-0 PerconaServerMongoDB is deprecated + and will be removed in v1.17.0; see v1.13.0 release notes for instructions to + migrate to psmdb.percona.com/v1 + name: v1-12-0 + schema: + openAPIV3Schema: + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + status: + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: false + subresources: + status: {} + - additionalPrinterColumns: + - jsonPath: .status.host + name: ENDPOINT + type: string + - jsonPath: .status.state + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1 + schema: + openAPIV3Schema: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + allowUnsafeConfigurations: + type: boolean + backup: + properties: + annotations: + additionalProperties: + type: string + type: object + configuration: + properties: + backupOptions: + properties: + oplogSpanMin: + type: number + priority: + additionalProperties: + type: number + type: object + timeouts: + properties: + startingStatus: + format: int32 + type: integer + type: object + required: + - oplogSpanMin + type: object + restoreOptions: + properties: + batchSize: + type: integer + downloadChunkMb: + type: integer + maxDownloadBufferMb: + type: integer + mongodLocation: + type: string + mongodLocationMap: + additionalProperties: + type: string + type: object + numDownloadWorkers: + type: integer + numInsertionWorkers: + type: integer + type: object + type: object + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + enabled: + type: boolean + image: + type: string + labels: + additionalProperties: + type: string + type: object + pitr: + properties: + compressionLevel: + type: integer + compressionType: + type: string + enabled: + type: boolean + oplogOnly: + type: boolean + oplogSpanMin: + type: number + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + storages: + additionalProperties: + properties: + azure: + properties: + container: + type: string + credentialsSecret: + type: string + endpointUrl: + type: string + prefix: + type: string + required: + - credentialsSecret + type: object + s3: + properties: + bucket: + type: string + credentialsSecret: + type: string + debugLogLevels: + type: string + endpointUrl: + type: string + forcePathStyle: + type: boolean + insecureSkipTLSVerify: + type: boolean + maxUploadParts: + type: integer + prefix: + type: string + region: + type: string + retryer: + properties: + maxRetryDelay: + type: string + minRetryDelay: + type: string + numMaxRetries: + type: integer + type: object + serverSideEncryption: + properties: + kmsKeyID: + type: string + sseAlgorithm: + type: string + sseCustomerAlgorithm: + type: string + sseCustomerKey: + type: string + type: object + storageClass: + type: string + uploadPartSize: + type: integer + required: + - bucket + type: object + type: + type: string + required: + - type + type: object + type: object + tasks: + items: + properties: + compressionLevel: + type: integer + compressionType: + type: string + enabled: + type: boolean + keep: + type: integer + name: + type: string + schedule: + type: string + storageName: + type: string + type: + enum: + - logical + - physical + type: string + required: + - enabled + - name + type: object + type: array + required: + - enabled + - image + type: object + clusterServiceDNSMode: + type: string + clusterServiceDNSSuffix: + type: string + crVersion: + type: string + enableVolumeExpansion: + type: boolean + ignoreAnnotations: + items: + type: string + type: array + ignoreLabels: + items: + type: string + type: array + image: + type: string + imagePullPolicy: + type: string + imagePullSecrets: + items: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + type: array + initContainerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + initImage: + type: string + multiCluster: + properties: + DNSSuffix: + type: string + enabled: + type: boolean + required: + - enabled + type: object + pause: + type: boolean + platform: + type: string + pmm: + properties: + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + enabled: + type: boolean + image: + type: string + mongodParams: + type: string + mongosParams: + type: string + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + serverHost: + type: string + required: + - image + type: object + replsets: + items: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + arbiter: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + enabled: + type: boolean + labels: + additionalProperties: + type: string + type: object + nodeSelector: + additionalProperties: + type: string + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + priorityClassName: + type: string + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + required: + - enabled + - size + type: object + clusterRole: + type: string + configuration: + type: string + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + expose: + properties: + annotations: + additionalProperties: + type: string + type: object + enabled: + type: boolean + exposeType: + type: string + externalTrafficPolicy: + type: string + internalTrafficPolicy: + type: string + labels: + additionalProperties: + type: string + type: object + loadBalancerSourceRanges: + items: + type: string + type: array + serviceAnnotations: + additionalProperties: + type: string + type: object + serviceLabels: + additionalProperties: + type: string + type: object + type: + type: string + required: + - enabled + type: object + externalNodes: + items: + properties: + horizons: + additionalProperties: + type: string + type: object + host: + type: string + port: + type: integer + priority: + type: integer + tags: + additionalProperties: + type: string + type: object + votes: + type: integer + required: + - host + - priority + - votes + type: object + type: array + hostAliases: + items: + properties: + hostnames: + items: + type: string + type: array + x-kubernetes-list-type: atomic + ip: + type: string + required: + - ip + type: object + type: array + labels: + additionalProperties: + type: string + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + startupDelaySeconds: + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + nodeSelector: + additionalProperties: + type: string + type: object + nonvoting: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + configuration: + type: string + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + enabled: + type: boolean + labels: + additionalProperties: + type: string + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + startupDelaySeconds: + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + nodeSelector: + additionalProperties: + type: string + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + priorityClassName: + type: string + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + volumeSpec: + properties: + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + persistentVolumeClaim: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + annotations: + additionalProperties: + type: string + type: object + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + labels: + additionalProperties: + type: string + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + type: object + required: + - enabled + - size + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + primaryPreferTagSelector: + additionalProperties: + type: string + type: object + priorityClassName: + type: string + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + replsetOverrides: + additionalProperties: + properties: + horizons: + additionalProperties: + type: string + type: object + host: + type: string + tags: + additionalProperties: + type: string + type: object + type: object + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + splitHorizons: + additionalProperties: + additionalProperties: + type: string + type: object + type: object + storage: + properties: + directoryPerDB: + type: boolean + engine: + type: string + inMemory: + properties: + engineConfig: + properties: + inMemorySizeRatio: + type: number + type: object + type: object + mmapv1: + properties: + nsSize: + type: integer + smallfiles: + type: boolean + type: object + syncPeriodSecs: + type: integer + wiredTiger: + properties: + collectionConfig: + properties: + blockCompressor: + type: string + type: object + engineConfig: + properties: + cacheSizeRatio: + type: number + directoryForIndexes: + type: boolean + journalCompressor: + type: string + type: object + indexConfig: + properties: + prefixCompression: + type: boolean + type: object + type: object + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + volumeSpec: + properties: + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + persistentVolumeClaim: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + annotations: + additionalProperties: + type: string + type: object + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + labels: + additionalProperties: + type: string + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + type: object + required: + - size + type: object + type: array + roles: + items: + properties: + authenticationRestrictions: + items: + properties: + clientSource: + items: + type: string + type: array + serverAddress: + items: + type: string + type: array + type: object + type: array + db: + type: string + privileges: + items: + properties: + actions: + items: + type: string + type: array + resource: + properties: + cluster: + type: boolean + collection: + type: string + db: + type: string + type: object + required: + - actions + type: object + type: array + role: + type: string + roles: + items: + properties: + db: + type: string + role: + type: string + required: + - db + - role + type: object + type: array + required: + - db + - privileges + - role + type: object + type: array + schedulerName: + type: string + secrets: + properties: + encryptionKey: + type: string + keyFile: + type: string + ldapSecret: + type: string + sse: + type: string + ssl: + type: string + sslInternal: + type: string + users: + type: string + vault: + type: string + type: object + sharding: + properties: + balancer: + properties: + enabled: + type: boolean + type: object + configsvrReplSet: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + arbiter: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + enabled: + type: boolean + labels: + additionalProperties: + type: string + type: object + nodeSelector: + additionalProperties: + type: string + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + priorityClassName: + type: string + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + required: + - enabled + - size + type: object + clusterRole: + type: string + configuration: + type: string + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + expose: + properties: + annotations: + additionalProperties: + type: string + type: object + enabled: + type: boolean + exposeType: + type: string + externalTrafficPolicy: + type: string + internalTrafficPolicy: + type: string + labels: + additionalProperties: + type: string + type: object + loadBalancerSourceRanges: + items: + type: string + type: array + serviceAnnotations: + additionalProperties: + type: string + type: object + serviceLabels: + additionalProperties: + type: string + type: object + type: + type: string + required: + - enabled + type: object + externalNodes: + items: + properties: + horizons: + additionalProperties: + type: string + type: object + host: + type: string + port: + type: integer + priority: + type: integer + tags: + additionalProperties: + type: string + type: object + votes: + type: integer + required: + - host + - priority + - votes + type: object + type: array + hostAliases: + items: + properties: + hostnames: + items: + type: string + type: array + x-kubernetes-list-type: atomic + ip: + type: string + required: + - ip + type: object + type: array + labels: + additionalProperties: + type: string + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + startupDelaySeconds: + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + nodeSelector: + additionalProperties: + type: string + type: object + nonvoting: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + configuration: + type: string + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + enabled: + type: boolean + labels: + additionalProperties: + type: string + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + startupDelaySeconds: + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + nodeSelector: + additionalProperties: + type: string + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + priorityClassName: + type: string + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + volumeSpec: + properties: + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + persistentVolumeClaim: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + annotations: + additionalProperties: + type: string + type: object + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + labels: + additionalProperties: + type: string + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + type: object + required: + - enabled + - size + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + primaryPreferTagSelector: + additionalProperties: + type: string + type: object + priorityClassName: + type: string + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + replsetOverrides: + additionalProperties: + properties: + horizons: + additionalProperties: + type: string + type: object + host: + type: string + tags: + additionalProperties: + type: string + type: object + type: object + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + splitHorizons: + additionalProperties: + additionalProperties: + type: string + type: object + type: object + storage: + properties: + directoryPerDB: + type: boolean + engine: + type: string + inMemory: + properties: + engineConfig: + properties: + inMemorySizeRatio: + type: number + type: object + type: object + mmapv1: + properties: + nsSize: + type: integer + smallfiles: + type: boolean + type: object + syncPeriodSecs: + type: integer + wiredTiger: + properties: + collectionConfig: + properties: + blockCompressor: + type: string + type: object + engineConfig: + properties: + cacheSizeRatio: + type: number + directoryForIndexes: + type: boolean + journalCompressor: + type: string + type: object + indexConfig: + properties: + prefixCompression: + type: boolean + type: object + type: object + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + volumeSpec: + properties: + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + persistentVolumeClaim: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + annotations: + additionalProperties: + type: string + type: object + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + labels: + additionalProperties: + type: string + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + type: object + required: + - size + type: object + enabled: + type: boolean + mongos: + properties: + affinity: + properties: + advanced: + properties: + nodeAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + preference: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + weight: + format: int32 + type: integer + required: + - preference + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + properties: + nodeSelectorTerms: + items: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchFields: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + type: object + x-kubernetes-map-type: atomic + type: array + x-kubernetes-list-type: atomic + required: + - nodeSelectorTerms + type: object + x-kubernetes-map-type: atomic + type: object + podAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + podAntiAffinity: + properties: + preferredDuringSchedulingIgnoredDuringExecution: + items: + properties: + podAffinityTerm: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + weight: + format: int32 + type: integer + required: + - podAffinityTerm + - weight + type: object + type: array + x-kubernetes-list-type: atomic + requiredDuringSchedulingIgnoredDuringExecution: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + mismatchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + namespaceSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + namespaces: + items: + type: string + type: array + x-kubernetes-list-type: atomic + topologyKey: + type: string + required: + - topologyKey + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + antiAffinityTopologyKey: + type: string + type: object + annotations: + additionalProperties: + type: string + type: object + configuration: + type: string + containerSecurityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + expose: + properties: + annotations: + additionalProperties: + type: string + type: object + exposeType: + type: string + externalTrafficPolicy: + type: string + internalTrafficPolicy: + type: string + labels: + additionalProperties: + type: string + type: object + loadBalancerSourceRanges: + items: + type: string + type: array + nodePort: + format: int32 + type: integer + serviceAnnotations: + additionalProperties: + type: string + type: object + serviceLabels: + additionalProperties: + type: string + type: object + servicePerPod: + type: boolean + type: + type: string + type: object + hostAliases: + items: + properties: + hostnames: + items: + type: string + type: array + x-kubernetes-list-type: atomic + ip: + type: string + required: + - ip + type: object + type: array + hostPort: + format: int32 + type: integer + labels: + additionalProperties: + type: string + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + startupDelaySeconds: + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + nodeSelector: + additionalProperties: + type: string + type: object + podDisruptionBudget: + properties: + maxUnavailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + minAvailable: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: object + podSecurityContext: + properties: + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + fsGroup: + format: int64 + type: integer + fsGroupChangePolicy: + type: string + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + supplementalGroups: + items: + format: int64 + type: integer + type: array + x-kubernetes-list-type: atomic + supplementalGroupsPolicy: + type: string + sysctls: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + port: + format: int32 + type: integer + priorityClassName: + type: string + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + runtimeClassName: + type: string + serviceAccountName: + type: string + setParameter: + properties: + cursorTimeoutMillis: + type: integer + type: object + sidecarPVCs: + items: + properties: + apiVersion: + type: string + kind: + type: string + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + status: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + allocatedResourceStatuses: + additionalProperties: + type: string + type: object + x-kubernetes-map-type: granular + allocatedResources: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + capacity: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + conditions: + items: + properties: + lastProbeTime: + format: date-time + type: string + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + currentVolumeAttributesClassName: + type: string + modifyVolumeStatus: + properties: + status: + type: string + targetVolumeAttributesClassName: + type: string + required: + - status + type: object + phase: + type: string + type: object + type: object + type: array + sidecarVolumes: + items: + properties: + awsElasticBlockStore: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + azureDisk: + properties: + cachingMode: + type: string + diskName: + type: string + diskURI: + type: string + fsType: + default: ext4 + type: string + kind: + type: string + readOnly: + default: false + type: boolean + required: + - diskName + - diskURI + type: object + azureFile: + properties: + readOnly: + type: boolean + secretName: + type: string + shareName: + type: string + required: + - secretName + - shareName + type: object + cephfs: + properties: + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + path: + type: string + readOnly: + type: boolean + secretFile: + type: string + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + type: string + required: + - monitors + type: object + cinder: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeID: + type: string + required: + - volumeID + type: object + configMap: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + csi: + properties: + driver: + type: string + fsType: + type: string + nodePublishSecretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + readOnly: + type: boolean + volumeAttributes: + additionalProperties: + type: string + type: object + required: + - driver + type: object + downwardAPI: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + emptyDir: + properties: + medium: + type: string + sizeLimit: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + ephemeral: + properties: + volumeClaimTemplate: + properties: + metadata: + type: object + spec: + properties: + accessModes: + items: + type: string + type: array + x-kubernetes-list-type: atomic + dataSource: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + required: + - kind + - name + type: object + x-kubernetes-map-type: atomic + dataSourceRef: + properties: + apiGroup: + type: string + kind: + type: string + name: + type: string + namespace: + type: string + required: + - kind + - name + type: object + resources: + properties: + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + selector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + storageClassName: + type: string + volumeAttributesClassName: + type: string + volumeMode: + type: string + volumeName: + type: string + type: object + required: + - spec + type: object + type: object + fc: + properties: + fsType: + type: string + lun: + format: int32 + type: integer + readOnly: + type: boolean + targetWWNs: + items: + type: string + type: array + x-kubernetes-list-type: atomic + wwids: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + flexVolume: + properties: + driver: + type: string + fsType: + type: string + options: + additionalProperties: + type: string + type: object + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + required: + - driver + type: object + flocker: + properties: + datasetName: + type: string + datasetUUID: + type: string + type: object + gcePersistentDisk: + properties: + fsType: + type: string + partition: + format: int32 + type: integer + pdName: + type: string + readOnly: + type: boolean + required: + - pdName + type: object + gitRepo: + properties: + directory: + type: string + repository: + type: string + revision: + type: string + required: + - repository + type: object + glusterfs: + properties: + endpoints: + type: string + path: + type: string + readOnly: + type: boolean + required: + - endpoints + - path + type: object + hostPath: + properties: + path: + type: string + type: + type: string + required: + - path + type: object + image: + properties: + pullPolicy: + type: string + reference: + type: string + type: object + iscsi: + properties: + chapAuthDiscovery: + type: boolean + chapAuthSession: + type: boolean + fsType: + type: string + initiatorName: + type: string + iqn: + type: string + iscsiInterface: + default: default + type: string + lun: + format: int32 + type: integer + portals: + items: + type: string + type: array + x-kubernetes-list-type: atomic + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + targetPortal: + type: string + required: + - iqn + - lun + - targetPortal + type: object + name: + type: string + nfs: + properties: + path: + type: string + readOnly: + type: boolean + server: + type: string + required: + - path + - server + type: object + persistentVolumeClaim: + properties: + claimName: + type: string + readOnly: + type: boolean + required: + - claimName + type: object + photonPersistentDisk: + properties: + fsType: + type: string + pdID: + type: string + required: + - pdID + type: object + portworxVolume: + properties: + fsType: + type: string + readOnly: + type: boolean + volumeID: + type: string + required: + - volumeID + type: object + projected: + properties: + defaultMode: + format: int32 + type: integer + sources: + items: + properties: + clusterTrustBundle: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + name: + type: string + optional: + type: boolean + path: + type: string + signerName: + type: string + required: + - path + type: object + configMap: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + downwardAPI: + properties: + items: + items: + properties: + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + mode: + format: int32 + type: integer + path: + type: string + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + required: + - path + type: object + type: array + x-kubernetes-list-type: atomic + type: object + secret: + properties: + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + serviceAccountToken: + properties: + audience: + type: string + expirationSeconds: + format: int64 + type: integer + path: + type: string + required: + - path + type: object + type: object + type: array + x-kubernetes-list-type: atomic + type: object + quobyte: + properties: + group: + type: string + readOnly: + type: boolean + registry: + type: string + tenant: + type: string + user: + type: string + volume: + type: string + required: + - registry + - volume + type: object + rbd: + properties: + fsType: + type: string + image: + type: string + keyring: + default: /etc/ceph/keyring + type: string + monitors: + items: + type: string + type: array + x-kubernetes-list-type: atomic + pool: + default: rbd + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + user: + default: admin + type: string + required: + - image + - monitors + type: object + scaleIO: + properties: + fsType: + default: xfs + type: string + gateway: + type: string + protectionDomain: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + sslEnabled: + type: boolean + storageMode: + default: ThinProvisioned + type: string + storagePool: + type: string + system: + type: string + volumeName: + type: string + required: + - gateway + - secretRef + - system + type: object + secret: + properties: + defaultMode: + format: int32 + type: integer + items: + items: + properties: + key: + type: string + mode: + format: int32 + type: integer + path: + type: string + required: + - key + - path + type: object + type: array + x-kubernetes-list-type: atomic + optional: + type: boolean + secretName: + type: string + type: object + storageos: + properties: + fsType: + type: string + readOnly: + type: boolean + secretRef: + properties: + name: + default: "" + type: string + type: object + x-kubernetes-map-type: atomic + volumeName: + type: string + volumeNamespace: + type: string + type: object + vsphereVolume: + properties: + fsType: + type: string + storagePolicyID: + type: string + storagePolicyName: + type: string + volumePath: + type: string + required: + - volumePath + type: object + required: + - name + type: object + type: array + sidecars: + items: + properties: + args: + items: + type: string + type: array + x-kubernetes-list-type: atomic + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + env: + items: + properties: + name: + type: string + value: + type: string + valueFrom: + properties: + configMapKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + properties: + apiVersion: + type: string + fieldPath: + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + properties: + containerName: + type: string + divisor: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + properties: + key: + type: string + name: + default: "" + type: string + optional: + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + envFrom: + items: + properties: + configMapRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + prefix: + type: string + secretRef: + properties: + name: + default: "" + type: string + optional: + type: boolean + type: object + x-kubernetes-map-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + image: + type: string + imagePullPolicy: + type: string + lifecycle: + properties: + postStart: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + preStop: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + sleep: + properties: + seconds: + format: int64 + type: integer + required: + - seconds + type: object + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: object + type: object + livenessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + name: + type: string + ports: + items: + properties: + containerPort: + format: int32 + type: integer + hostIP: + type: string + hostPort: + format: int32 + type: integer + name: + type: string + protocol: + default: TCP + type: string + required: + - containerPort + type: object + type: array + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + resizePolicy: + items: + properties: + resourceName: + type: string + restartPolicy: + type: string + required: + - resourceName + - restartPolicy + type: object + type: array + x-kubernetes-list-type: atomic + resources: + properties: + claims: + items: + properties: + name: + type: string + request: + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + requests: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + restartPolicy: + type: string + securityContext: + properties: + allowPrivilegeEscalation: + type: boolean + appArmorProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + capabilities: + properties: + add: + items: + type: string + type: array + x-kubernetes-list-type: atomic + drop: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + privileged: + type: boolean + procMount: + type: string + readOnlyRootFilesystem: + type: boolean + runAsGroup: + format: int64 + type: integer + runAsNonRoot: + type: boolean + runAsUser: + format: int64 + type: integer + seLinuxOptions: + properties: + level: + type: string + role: + type: string + type: + type: string + user: + type: string + type: object + seccompProfile: + properties: + localhostProfile: + type: string + type: + type: string + required: + - type + type: object + windowsOptions: + properties: + gmsaCredentialSpec: + type: string + gmsaCredentialSpecName: + type: string + hostProcess: + type: boolean + runAsUserName: + type: string + type: object + type: object + startupProbe: + properties: + exec: + properties: + command: + items: + type: string + type: array + x-kubernetes-list-type: atomic + type: object + failureThreshold: + format: int32 + type: integer + grpc: + properties: + port: + format: int32 + type: integer + service: + default: "" + type: string + required: + - port + type: object + httpGet: + properties: + host: + type: string + httpHeaders: + items: + properties: + name: + type: string + value: + type: string + required: + - name + - value + type: object + type: array + x-kubernetes-list-type: atomic + path: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + type: string + required: + - port + type: object + initialDelaySeconds: + format: int32 + type: integer + periodSeconds: + format: int32 + type: integer + successThreshold: + format: int32 + type: integer + tcpSocket: + properties: + host: + type: string + port: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + terminationGracePeriodSeconds: + format: int64 + type: integer + timeoutSeconds: + format: int32 + type: integer + type: object + stdin: + type: boolean + stdinOnce: + type: boolean + terminationMessagePath: + type: string + terminationMessagePolicy: + type: string + tty: + type: boolean + volumeDevices: + items: + properties: + devicePath: + type: string + name: + type: string + required: + - devicePath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - devicePath + x-kubernetes-list-type: map + volumeMounts: + items: + properties: + mountPath: + type: string + mountPropagation: + type: string + name: + type: string + readOnly: + type: boolean + recursiveReadOnly: + type: string + subPath: + type: string + subPathExpr: + type: string + required: + - mountPath + - name + type: object + type: array + x-kubernetes-list-map-keys: + - mountPath + x-kubernetes-list-type: map + workingDir: + type: string + required: + - name + type: object + type: array + size: + format: int32 + type: integer + terminationGracePeriodSeconds: + format: int64 + type: integer + tolerations: + items: + properties: + effect: + type: string + key: + type: string + operator: + type: string + tolerationSeconds: + format: int64 + type: integer + value: + type: string + type: object + type: array + topologySpreadConstraints: + items: + properties: + labelSelector: + properties: + matchExpressions: + items: + properties: + key: + type: string + operator: + type: string + values: + items: + type: string + type: array + x-kubernetes-list-type: atomic + required: + - key + - operator + type: object + type: array + x-kubernetes-list-type: atomic + matchLabels: + additionalProperties: + type: string + type: object + type: object + x-kubernetes-map-type: atomic + matchLabelKeys: + items: + type: string + type: array + x-kubernetes-list-type: atomic + maxSkew: + format: int32 + type: integer + minDomains: + format: int32 + type: integer + nodeAffinityPolicy: + type: string + nodeTaintsPolicy: + type: string + topologyKey: + type: string + whenUnsatisfiable: + type: string + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + type: object + type: array + type: object + required: + - enabled + type: object + tls: + properties: + allowInvalidCertificates: + type: boolean + certValidityDuration: + type: string + issuerConf: + properties: + group: + type: string + kind: + type: string + name: + type: string + required: + - name + type: object + mode: + type: string + type: object + unmanaged: + type: boolean + unsafeFlags: + properties: + backupIfUnhealthy: + type: boolean + mongosSize: + type: boolean + replsetSize: + type: boolean + terminationGracePeriod: + type: boolean + tls: + type: boolean + type: object + updateStrategy: + type: string + upgradeOptions: + properties: + apply: + type: string + schedule: + type: string + setFCV: + type: boolean + versionServiceEndpoint: + type: string + type: object + users: + items: + properties: + db: + type: string + name: + type: string + passwordSecretRef: + properties: + key: + type: string + name: + type: string + required: + - name + type: object + roles: + items: + properties: + db: + type: string + name: + type: string + required: + - db + - name + type: object + type: array + required: + - name + - passwordSecretRef + - roles + type: object + type: array + required: + - image + type: object + status: + properties: + backup: + type: string + backupVersion: + type: string + conditions: + items: + properties: + lastTransitionTime: + format: date-time + type: string + message: + type: string + reason: + type: string + status: + type: string + type: + type: string + required: + - status + - type + type: object + type: array + host: + type: string + message: + type: string + mongoImage: + type: string + mongoVersion: + type: string + mongos: + properties: + message: + type: string + ready: + type: integer + size: + type: integer + status: + type: string + required: + - ready + - size + type: object + observedGeneration: + format: int64 + type: integer + pmmStatus: + type: string + pmmVersion: + type: string + ready: + format: int32 + type: integer + replsets: + additionalProperties: + properties: + added_as_shard: + type: boolean + clusterRole: + type: string + initialized: + type: boolean + members: + items: + properties: + name: + type: string + version: + type: string + type: object + type: array + message: + type: string + ready: + format: int32 + type: integer + size: + format: int32 + type: integer + status: + type: string + required: + - ready + - size + type: object + type: object + size: + format: int32 + type: integer + state: + type: string + required: + - ready + - size + type: object + type: object + served: true + storage: true + subresources: + status: {} \ No newline at end of file diff --git a/charts/percona/psmdb-operator/1.18.0/templates/NOTES.txt b/charts/percona/psmdb-operator/1.18.0/templates/NOTES.txt new file mode 100644 index 0000000000..f72a33c0df --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/NOTES.txt @@ -0,0 +1,15 @@ +1. Percona Operator for MongoDB is deployed. + See if the operator Pod is running: + + kubectl get pods -l app.kubernetes.io/name=psmdb-operator --namespace {{ .Release.Namespace }} + + Check the operator logs if the Pod is not starting: + + export POD=$(kubectl get pods -l app.kubernetes.io/name=psmdb-operator --namespace {{ .Release.Namespace }} --output name) + kubectl logs $POD --namespace={{ .Release.Namespace }} + +2. Deploy the database cluster from psmdb-db chart: + + helm install my-db percona/psmdb-db --namespace={{ .Release.Namespace }} + +Read more in our documentation: https://docs.percona.com/percona-operator-for-mongodb/ diff --git a/charts/percona/psmdb-operator/1.18.0/templates/_helpers.tpl b/charts/percona/psmdb-operator/1.18.0/templates/_helpers.tpl new file mode 100644 index 0000000000..1bf81ed17d --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/_helpers.tpl @@ -0,0 +1,45 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "psmdb-operator.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If release name contains chart name it will be used as a full name. +*/}} +{{- define "psmdb-operator.fullname" -}} +{{- if .Values.fullnameOverride -}} +{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "psmdb-operator.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Common labels +*/}} +{{- define "psmdb-operator.labels" -}} +app.kubernetes.io/name: {{ include "psmdb-operator.name" . }} +helm.sh/chart: {{ include "psmdb-operator.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} diff --git a/charts/percona/psmdb-operator/1.18.0/templates/deployment.yaml b/charts/percona/psmdb-operator/1.18.0/templates/deployment.yaml new file mode 100644 index 0000000000..41f26c2646 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/deployment.yaml @@ -0,0 +1,108 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "psmdb-operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "psmdb-operator.labels" . | nindent 4 }} + {{- with .Values.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + replicas: {{ .Values.replicaCount }} + selector: + matchLabels: + app.kubernetes.io/name: {{ include "psmdb-operator.name" . }} + app.kubernetes.io/instance: {{ .Release.Name }} + template: + metadata: + {{- with .Values.podAnnotations }} + annotations: + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + app.kubernetes.io/name: {{ include "psmdb-operator.name" . }} + app.kubernetes.io/instance: {{ .Release.Name }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + serviceAccountName: {{ include "psmdb-operator.fullname" . }} + securityContext: + {{- toYaml .Values.podSecurityContext | nindent 8 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - name: {{ .Chart.Name }} + securityContext: + {{- toYaml .Values.securityContext | nindent 12 }} + image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + ports: + - containerPort: 8080 + protocol: TCP + name: metrics + - containerPort: 8081 + protocol: TCP + name: health + command: + - percona-server-mongodb-operator + {{- if .Values.securityContext.readOnlyRootFilesystem }} + volumeMounts: + - name: tmpdir + mountPath: /tmp + {{- end }} + env: + - name: LOG_STRUCTURED + value: "{{ .Values.logStructured }}" + - name: LOG_LEVEL + value: "{{ .Values.logLevel }}" + - name: WATCH_NAMESPACE + {{- if .Values.watchAllNamespaces }} + value: "" + {{- else }} + value: "{{ default .Release.Namespace .Values.watchNamespace }}" + {{- end }} + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: OPERATOR_NAME + value: {{ default "percona-server-mongodb-operator" .Values.operatorName }} + - name: RESYNC_PERIOD + value: "{{ .Values.env.resyncPeriod }}" + - name: DISABLE_TELEMETRY + value: "{{ .Values.disableTelemetry }}" + livenessProbe: + httpGet: + path: /healthz + port: health + readinessProbe: + httpGet: + path: /healthz + port: health + resources: + {{- toYaml .Values.resources | nindent 12 }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.securityContext.readOnlyRootFilesystem }} + volumes: + - name: tmpdir + emptyDir: {} + {{- end }} diff --git a/charts/percona/psmdb-operator/1.18.0/templates/namespace.yaml b/charts/percona/psmdb-operator/1.18.0/templates/namespace.yaml new file mode 100644 index 0000000000..cfc96d4d9b --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/namespace.yaml @@ -0,0 +1,11 @@ +{{ if and .Values.watchNamespace .Values.createNamespace }} +{{ range ( split "," .Values.watchNamespace ) }} +apiVersion: v1 +kind: Namespace +metadata: + name: {{ trim . }} + annotations: + helm.sh/resource-policy: keep +--- +{{ end }} +{{ end }} diff --git a/charts/percona/psmdb-operator/1.18.0/templates/role-binding.yaml b/charts/percona/psmdb-operator/1.18.0/templates/role-binding.yaml new file mode 100644 index 0000000000..a815869d45 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/role-binding.yaml @@ -0,0 +1,41 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "psmdb-operator.fullname" . }} + namespace: {{ .Release.Namespace }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +--- +{{- end }} +{{- if .Values.rbac.create }} +{{- if or .Values.watchNamespace .Values.watchAllNamespaces }} +kind: ClusterRoleBinding +{{- else }} +kind: RoleBinding +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: service-account-{{ include "psmdb-operator.fullname" . }} +{{- if not (or .Values.watchNamespace .Values.watchAllNamespaces) }} + namespace: {{ .Release.Namespace }} +{{- end }} + labels: +{{ include "psmdb-operator.labels" . | indent 4 }} +subjects: +- kind: ServiceAccount + name: {{ include "psmdb-operator.fullname" . }} + {{- if or .Values.watchNamespace .Values.watchAllNamespaces }} + namespace: {{ .Release.Namespace }} + {{- end }} +roleRef: + {{- if or .Values.watchNamespace .Values.watchAllNamespaces }} + kind: ClusterRole + {{- else }} + kind: Role + {{- end }} + name: {{ include "psmdb-operator.fullname" . }} + apiGroup: rbac.authorization.k8s.io +{{- end }} diff --git a/charts/percona/psmdb-operator/1.18.0/templates/role.yaml b/charts/percona/psmdb-operator/1.18.0/templates/role.yaml new file mode 100644 index 0000000000..4d65e6a7a7 --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/templates/role.yaml @@ -0,0 +1,167 @@ +{{- if .Values.rbac.create }} +{{- if or .Values.watchNamespace .Values.watchAllNamespaces }} +kind: ClusterRole +{{- else }} +kind: Role +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "psmdb-operator.fullname" . }} +{{- if not (or .Values.watchNamespace .Values.watchAllNamespaces) }} + namespace: {{ .Release.Namespace }} +{{- end }} + labels: +{{ include "psmdb-operator.labels" . | indent 4 }} +rules: + - apiGroups: + - psmdb.percona.com + resources: + - perconaservermongodbs + - perconaservermongodbs/status + - perconaservermongodbs/finalizers + - perconaservermongodbbackups + - perconaservermongodbbackups/status + - perconaservermongodbbackups/finalizers + - perconaservermongodbrestores + - perconaservermongodbrestores/status + - perconaservermongodbrestores/finalizers + verbs: + - get + - list + - watch + - create + - update + - patch + - delete +{{- if or .Values.watchNamespace .Values.watchAllNamespaces }} + - apiGroups: + - admissionregistration.k8s.io + resources: + - validatingwebhookconfigurations + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - "" + resources: + - nodes + verbs: + - get + - list + - watch +{{- end }} + - apiGroups: + - "" + resources: + - pods + - pods/exec + - services + - persistentvolumeclaims + - secrets + - configmaps + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - apps + resources: + - deployments + - replicasets + - statefulsets + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - batch + resources: + - cronjobs + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - policy + resources: + - poddisruptionbudgets + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - events.k8s.io + - "" + resources: + - events + verbs: + - get + - list + - watch + - create + - patch + - apiGroups: + - certmanager.k8s.io + - cert-manager.io + resources: + - issuers + - certificates + - certificaterequests + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - deletecollection + - apiGroups: + - net.gke.io + - multicluster.x-k8s.io + resources: + - serviceexports + - serviceimports + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - deletecollection +{{- end }} diff --git a/charts/percona/psmdb-operator/1.18.0/values.yaml b/charts/percona/psmdb-operator/1.18.0/values.yaml new file mode 100644 index 0000000000..e954e8f26e --- /dev/null +++ b/charts/percona/psmdb-operator/1.18.0/values.yaml @@ -0,0 +1,99 @@ +# Default values for psmdb-operator. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +replicaCount: 1 + +image: + repository: percona/percona-server-mongodb-operator + tag: 1.18.0 + pullPolicy: IfNotPresent + +# disableTelemetry: according to +# https://docs.percona.com/percona-operator-for-mongodb/telemetry.html +# this is how you can disable telemetry collection +# default is false which means telemetry will be collected +disableTelemetry: false + +# set if you want to specify a namespace to watch +# defaults to `.Release.namespace` if left blank +# multiple namespaces can be specified and separated by comma +# watchNamespace: +# set if you want that watched namespaces are created by helm +# createNamespace: false + +# set if operator should be deployed in cluster wide mode. defaults to false +watchAllNamespaces: false + +# rbac: settings for deployer RBAC creation +rbac: + # rbac.create: if false RBAC resources should be in place + create: true + +# serviceAccount: settings for Service Accounts used by the deployer +serviceAccount: + # serviceAccount.create: Whether to create the Service Accounts or not + create: true + # annotations to add to the service account + annotations: {} + +# annotations to add to the operator deployment +annotations: {} + +# labels to add to the operator deployment +labels: {} + +# annotations to add to the operator pod +podAnnotations: {} + # prometheus.io/scrape: "true" + # prometheus.io/port: "8080" + +# labels to the operator pod +podLabels: {} + +podSecurityContext: {} + # runAsNonRoot: true + # runAsUser: 2 + # runAsGroup: 2 + # fsGroup: 2 + # fsGroupChangePolicy: "OnRootMismatch" + +securityContext: {} + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - ALL + # seccompProfile: + # type: RuntimeDefault + +# set if you want to use a different operator name +# defaults to `percona-server-mongodb-operator` +# operatorName: + +imagePullSecrets: [] +nameOverride: "" +fullnameOverride: "" + +env: + resyncPeriod: 5s + +resources: {} + # We usually recommend not to specify default resources and to leave this as a conscious + # choice for the user. This also increases chances charts run on environments with little + # resources, such as Minikube. If you do want to specify resources, uncomment the following + # lines, adjust them as necessary, and remove the curly braces after 'resources:'. + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +nodeSelector: {} + +tolerations: [] + +affinity: {} + +logStructured: false +logLevel: "INFO" diff --git a/charts/redpanda/redpanda/5.9.10/.helmignore b/charts/redpanda/redpanda/5.9.10/.helmignore new file mode 100644 index 0000000000..d5bb5e6ba6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/.helmignore @@ -0,0 +1,28 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +README.md.gotmpl +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +*.go +testdata/ +ci/ diff --git a/charts/redpanda/redpanda/5.9.10/Chart.lock b/charts/redpanda/redpanda/5.9.10/Chart.lock new file mode 100644 index 0000000000..9b0e72b48f --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/Chart.lock @@ -0,0 +1,9 @@ +dependencies: +- name: console + repository: https://charts.redpanda.com + version: 0.7.30 +- name: connectors + repository: https://charts.redpanda.com + version: 0.1.13 +digest: sha256:f40126de897d3ec8d707b28408763bfad218f7c688575371f6fa3cb371eb884c +generated: "2024-10-16T12:27:15.358101+02:00" diff --git a/charts/redpanda/redpanda/5.9.10/Chart.yaml b/charts/redpanda/redpanda/5.9.10/Chart.yaml new file mode 100644 index 0000000000..be6aeaab3a --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/Chart.yaml @@ -0,0 +1,38 @@ +annotations: + artifacthub.io/images: | + - name: redpanda + image: docker.redpanda.com/redpandadata/redpanda:v24.2.7 + - name: busybox + image: busybox:latest + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Documentation + url: https://docs.redpanda.com + - name: "Helm (>= 3.10.0)" + url: https://helm.sh/docs/intro/install/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Redpanda + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: redpanda +apiVersion: v2 +appVersion: v24.2.7 +dependencies: +- condition: console.enabled + name: console + repository: https://charts.redpanda.com + version: '>=0.5 <1.0' +- condition: connectors.enabled + name: connectors + repository: https://charts.redpanda.com + version: '>=0.1.2 <1.0' +description: Redpanda is the real-time engine for modern apps. +icon: file://assets/icons/redpanda.svg +kubeVersion: '>=1.21-0' +maintainers: +- name: redpanda-data + url: https://github.com/orgs/redpanda-data/people +name: redpanda +sources: +- https://github.com/redpanda-data/helm-charts +type: application +version: 5.9.10 diff --git a/charts/redpanda/redpanda/5.9.10/LICENSE b/charts/redpanda/redpanda/5.9.10/LICENSE new file mode 100644 index 0000000000..261eeb9e9f --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/charts/redpanda/redpanda/5.9.10/README.md b/charts/redpanda/redpanda/5.9.10/README.md new file mode 100644 index 0000000000..52523aed5a --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/README.md @@ -0,0 +1,1220 @@ +# Redpanda Helm Chart Specification +--- +description: Find the default values and descriptions of settings in the Redpanda Helm chart. +--- + +![Version: 5.9.10](https://img.shields.io/badge/Version-5.9.10-informational?style=flat-square) ![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![AppVersion: v24.2.7](https://img.shields.io/badge/AppVersion-v24.2.7-informational?style=flat-square) + +This page describes the official Redpanda Helm Chart. In particular, this page describes the contents of the chart’s [`values.yaml` file](https://github.com/redpanda-data/helm-charts/blob/main/charts/redpanda/values.yaml). Each of the settings is listed and described on this page, along with any default values. + +For instructions on how to install and use the chart, including how to override and customize the chart’s values, refer to the [deployment documentation](https://docs.redpanda.com/docs/deploy/deployment-option/self-hosted/kubernetes/kubernetes-deploy/). + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.13.1](https://github.com/norwoodj/helm-docs/releases/v1.13.1) + +## Source Code + +* + +## Requirements + +Kubernetes: `>= 1.25.0-0` + +| Repository | Name | Version | +|------------|------|---------| +| https://charts.redpanda.com | connectors | >=0.1.2 <1.0 | +| https://charts.redpanda.com | console | >=0.5 <1.0 | + +## Settings + +### [affinity](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=affinity) + +Affinity constraints for scheduling Pods, can override this for StatefulSets and Jobs. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#affinity-and-anti-affinity). + +**Default:** `{}` + +### [auditLogging](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging) + +Audit logging for a redpanda cluster, must have enabled sasl and have one kafka listener supporting sasl authentication for audit logging to work. Note this feature is only available for redpanda versions >= v23.3.0. + +**Default:** + +``` +{"clientMaxBufferSize":16777216,"enabled":false,"enabledEventTypes":null,"excludedPrincipals":null,"excludedTopics":null,"listener":"internal","partitions":12,"queueDrainIntervalMs":500,"queueMaxBufferSizePerShard":1048576,"replicationFactor":null} +``` + +### [auditLogging.clientMaxBufferSize](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.clientMaxBufferSize) + +Defines the number of bytes (in bytes) allocated by the internal audit client for audit messages. + +**Default:** `16777216` + +### [auditLogging.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.enabled) + +Enable or disable audit logging, for production clusters we suggest you enable, however, this will only work if you also enable sasl and a listener with sasl enabled. + +**Default:** `false` + +### [auditLogging.enabledEventTypes](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.enabledEventTypes) + +Event types that should be captured by audit logs, default is [`admin`, `authenticate`, `management`]. + +**Default:** `nil` + +### [auditLogging.excludedPrincipals](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.excludedPrincipals) + +List of principals to exclude from auditing, default is null. + +**Default:** `nil` + +### [auditLogging.excludedTopics](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.excludedTopics) + +List of topics to exclude from auditing, default is null. + +**Default:** `nil` + +### [auditLogging.listener](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.listener) + +Kafka listener name, note that it must have `authenticationMethod` set to `sasl`. For external listeners, use the external listener name, such as `default`. + +**Default:** `"internal"` + +### [auditLogging.partitions](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.partitions) + +Integer value defining the number of partitions used by a newly created audit topic. + +**Default:** `12` + +### [auditLogging.queueDrainIntervalMs](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.queueDrainIntervalMs) + +In ms, frequency in which per shard audit logs are batched to client for write to audit log. + +**Default:** `500` + +### [auditLogging.queueMaxBufferSizePerShard](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.queueMaxBufferSizePerShard) + +Defines the maximum amount of memory used (in bytes) by the audit buffer in each shard. + +**Default:** `1048576` + +### [auditLogging.replicationFactor](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auditLogging.replicationFactor) + +Defines the replication factor for a newly created audit log topic. This configuration applies only to the audit log topic and may be different from the cluster or other topic configurations. This cannot be altered for existing audit log topics. Setting this value is optional. If a value is not provided, Redpanda will use the `internal_topic_replication_factor cluster` config value. Default is `null` + +**Default:** `nil` + +### [auth](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth) + +Authentication settings. For details, see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/). + +**Default:** + +``` +{"sasl":{"bootstrapUser":{"mechanism":"SCRAM-SHA-256"},"enabled":false,"mechanism":"SCRAM-SHA-512","secretRef":"redpanda-users","users":[]}} +``` + +### [auth.sasl.bootstrapUser](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.bootstrapUser) + +Details about how to create the bootstrap user for the cluster. The secretKeyRef is optionally specified. If it is specified, the chart will use a password written to that secret when creating the "kubernetes-controller" bootstrap user. If it is unspecified, then the secret will be generated and stored in the secret "releasename"-bootstrap-user, with the key "password". + +**Default:** + +``` +{"mechanism":"SCRAM-SHA-256"} +``` + +### [auth.sasl.bootstrapUser.mechanism](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.bootstrapUser.mechanism) + +The authentication mechanism to use for the bootstrap user. Options are `SCRAM-SHA-256` and `SCRAM-SHA-512`. + +**Default:** `"SCRAM-SHA-256"` + +### [auth.sasl.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.enabled) + +Enable SASL authentication. If you enable SASL authentication, you must provide a Secret in `auth.sasl.secretRef`. + +**Default:** `false` + +### [auth.sasl.mechanism](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.mechanism) + +The authentication mechanism to use for the superuser. Options are `SCRAM-SHA-256` and `SCRAM-SHA-512`. + +**Default:** `"SCRAM-SHA-512"` + +### [auth.sasl.secretRef](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.secretRef) + +A Secret that contains your superuser credentials. For details, see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/#use-secrets). + +**Default:** `"redpanda-users"` + +### [auth.sasl.users](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=auth.sasl.users) + +Optional list of superusers. These superusers will be created in the Secret whose name is defined in `auth.sasl.secretRef`. If this list is empty, the Secret in `auth.sasl.secretRef` must already exist in the cluster before you deploy the chart. Uncomment the sample list if you wish to try adding sample sasl users or override to use your own. + +**Default:** `[]` + +### [clusterDomain](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=clusterDomain) + +Default Kubernetes cluster domain. + +**Default:** `"cluster.local"` + +### [commonLabels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=commonLabels) + +Additional labels to add to all Kubernetes objects. For example, `my.k8s.service: redpanda`. + +**Default:** `{}` + +### [config](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config) + +This section contains various settings supported by Redpanda that may not work correctly in a Kubernetes cluster. Changing these settings comes with some risk. Use these settings to customize various Redpanda configurations that are not covered in other sections. These values have no impact on the configuration or behavior of the Kubernetes objects deployed by Helm, and therefore should not be modified for the purpose of configuring those objects. Instead, these settings get passed directly to the Redpanda binary at startup. For descriptions of these properties, see the [configuration documentation](https://docs.redpanda.com/docs/cluster-administration/configuration/). + +**Default:** + +``` +{"cluster":{},"node":{"crash_loop_limit":5},"pandaproxy_client":{},"rpk":{},"schema_registry_client":{},"tunable":{"compacted_log_segment_size":67108864,"kafka_connection_rate_limit":1000,"log_segment_size_max":268435456,"log_segment_size_min":16777216,"max_compacted_log_segment_size":536870912}} +``` + +### [config.cluster](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.cluster) + +[Cluster Configuration Properties](https://docs.redpanda.com/current/reference/properties/cluster-properties/) + +**Default:** `{}` + +### [config.node](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.node) + +[Broker (node) Configuration Properties](https://docs.redpanda.com/docs/reference/broker-properties/). + +**Default:** `{"crash_loop_limit":5}` + +### [config.node.crash_loop_limit](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.node.crash_loop_limit) + +Crash loop limit A limit on the number of consecutive times a broker can crash within one hour before its crash-tracking logic is reset. This limit prevents a broker from getting stuck in an infinite cycle of crashes. User can disable this crash loop limit check by the following action: * One hour elapses since the last crash * The node configuration file, redpanda.yaml, is updated via config.cluster or config.node or config.tunable objects * The startup_log file in the node’s data_directory is manually deleted Default to 5 REF: https://docs.redpanda.com/current/reference/broker-properties/#crash_loop_limit + +**Default:** `5` + +### [config.tunable](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable) + +Tunable cluster properties. Deprecated: all settings here may be specified via `config.cluster`. + +**Default:** + +``` +{"compacted_log_segment_size":67108864,"kafka_connection_rate_limit":1000,"log_segment_size_max":268435456,"log_segment_size_min":16777216,"max_compacted_log_segment_size":536870912} +``` + +### [config.tunable.compacted_log_segment_size](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable.compacted_log_segment_size) + +See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#compacted_log_segment_size). + +**Default:** `67108864` + +### [config.tunable.kafka_connection_rate_limit](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable.kafka_connection_rate_limit) + +See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#kafka_connection_rate_limit). + +**Default:** `1000` + +### [config.tunable.log_segment_size_max](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable.log_segment_size_max) + +See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#log_segment_size_max). + +**Default:** `268435456` + +### [config.tunable.log_segment_size_min](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable.log_segment_size_min) + +See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#log_segment_size_min). + +**Default:** `16777216` + +### [config.tunable.max_compacted_log_segment_size](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=config.tunable.max_compacted_log_segment_size) + +See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#max_compacted_log_segment_size). + +**Default:** `536870912` + +### [connectors](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=connectors) + +Redpanda Managed Connectors settings For a reference of configuration settings, see the [Redpanda Connectors documentation](https://docs.redpanda.com/docs/deploy/deployment-option/cloud/managed-connectors/). + +**Default:** + +``` +{"deployment":{"create":false},"enabled":false,"test":{"create":false}} +``` + +### [console](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=console) + +Redpanda Console settings. For a reference of configuration settings, see the [Redpanda Console documentation](https://docs.redpanda.com/docs/reference/console/config/). + +**Default:** + +``` +{"config":{},"configmap":{"create":false},"deployment":{"create":false},"enabled":true,"secret":{"create":false}} +``` + +### [enterprise](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=enterprise) + +Enterprise (optional) For details, see the [License documentation](https://docs.redpanda.com/docs/get-started/licenses/?platform=kubernetes#redpanda-enterprise-edition). + +**Default:** + +``` +{"license":"","licenseSecretRef":{}} +``` + +### [enterprise.license](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=enterprise.license) + +license (optional). + +**Default:** `""` + +### [enterprise.licenseSecretRef](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=enterprise.licenseSecretRef) + +Secret name and key where the license key is stored. + +**Default:** `{}` + +### [external](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=external) + +External access settings. For details, see the [Networking and Connectivity documentation](https://docs.redpanda.com/docs/manage/kubernetes/networking/networking-and-connectivity/). + +**Default:** + +``` +{"enabled":true,"service":{"enabled":true},"type":"NodePort"} +``` + +### [external.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=external.enabled) + +Enable external access for each Service. You can toggle external access for each listener in `listeners..external..enabled`. + +**Default:** `true` + +### [external.service](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=external.service) + +Service allows you to manage the creation of an external kubernetes service object + +**Default:** `{"enabled":true}` + +### [external.service.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=external.service.enabled) + +Enabled if set to false will not create the external service type You can still set your cluster with external access but not create the supporting service (NodePort/LoadBalander). Set this to false if you rather manage your own service. + +**Default:** `true` + +### [external.type](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=external.type) + +External access type. Only `NodePort` and `LoadBalancer` are supported. If undefined, then advertised listeners will be configured in Redpanda, but the helm chart will not create a Service. You must create a Service manually. Warning: If you use LoadBalancers, you will likely experience higher latency and increased packet loss. NodePort is recommended in cases where latency is a priority. + +**Default:** `"NodePort"` + +### [fullnameOverride](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=fullnameOverride) + +Override `redpanda.fullname` template. + +**Default:** `""` + +### [image](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=image) + +Redpanda Docker image settings. + +**Default:** + +``` +{"pullPolicy":"IfNotPresent","repository":"docker.redpanda.com/redpandadata/redpanda","tag":""} +``` + +### [image.pullPolicy](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=image.pullPolicy) + +The imagePullPolicy. If `image.tag` is 'latest', the default is `Always`. + +**Default:** `"IfNotPresent"` + +### [image.repository](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=image.repository) + +Docker repository from which to pull the Redpanda Docker image. + +**Default:** + +``` +"docker.redpanda.com/redpandadata/redpanda" +``` + +### [image.tag](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=image.tag) + +The Redpanda version. See DockerHub for: [All stable versions](https://hub.docker.com/r/redpandadata/redpanda/tags) and [all unstable versions](https://hub.docker.com/r/redpandadata/redpanda-unstable/tags). + +**Default:** `Chart.appVersion`. + +### [imagePullSecrets](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=imagePullSecrets) + +Pull secrets may be used to provide credentials to image repositories See the [Kubernetes documentation](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). + +**Default:** `[]` + +### [license_key](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=license_key) + +DEPRECATED Enterprise license key (optional). For details, see the [License documentation](https://docs.redpanda.com/docs/get-started/licenses/?platform=kubernetes#redpanda-enterprise-edition). + +**Default:** `""` + +### [license_secret_ref](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=license_secret_ref) + +DEPRECATED Secret name and secret key where the license key is stored. + +**Default:** `{}` + +### [listeners](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners) + +Listener settings. Override global settings configured above for individual listeners. For details, see the [listeners documentation](https://docs.redpanda.com/docs/manage/kubernetes/networking/configure-listeners/). + +**Default:** + +``` +{"admin":{"external":{"default":{"advertisedPorts":[31644],"port":9645,"tls":{"cert":"external"}}},"port":9644,"tls":{"cert":"default","requireClientAuth":false}},"http":{"authenticationMethod":null,"enabled":true,"external":{"default":{"advertisedPorts":[30082],"authenticationMethod":null,"port":8083,"tls":{"cert":"external","requireClientAuth":false}}},"kafkaEndpoint":"default","port":8082,"tls":{"cert":"default","requireClientAuth":false}},"kafka":{"authenticationMethod":null,"external":{"default":{"advertisedPorts":[31092],"authenticationMethod":null,"port":9094,"tls":{"cert":"external"}}},"port":9093,"tls":{"cert":"default","requireClientAuth":false}},"rpc":{"port":33145,"tls":{"cert":"default","requireClientAuth":false}},"schemaRegistry":{"authenticationMethod":null,"enabled":true,"external":{"default":{"advertisedPorts":[30081],"authenticationMethod":null,"port":8084,"tls":{"cert":"external","requireClientAuth":false}}},"kafkaEndpoint":"default","port":8081,"tls":{"cert":"default","requireClientAuth":false}}} +``` + +### [listeners.admin](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin) + +Admin API listener (only one). + +**Default:** + +``` +{"external":{"default":{"advertisedPorts":[31644],"port":9645,"tls":{"cert":"external"}}},"port":9644,"tls":{"cert":"default","requireClientAuth":false}} +``` + +### [listeners.admin.external](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.external) + +Optional external access settings. + +**Default:** + +``` +{"default":{"advertisedPorts":[31644],"port":9645,"tls":{"cert":"external"}}} +``` + +### [listeners.admin.external.default](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.external.default) + +Name of the external listener. + +**Default:** + +``` +{"advertisedPorts":[31644],"port":9645,"tls":{"cert":"external"}} +``` + +### [listeners.admin.external.default.tls](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.external.default.tls) + +The port advertised to this listener's external clients. List one port if you want to use the same port for each broker (would be the case when using NodePort service). Otherwise, list the port you want to use for each broker in order of StatefulSet replicas. If undefined, `listeners.admin.port` is used. + +**Default:** `{"cert":"external"}` + +### [listeners.admin.port](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.port) + +The port for both internal and external connections to the Admin API. + +**Default:** `9644` + +### [listeners.admin.tls](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.tls) + +Optional TLS section (required if global TLS is enabled) + +**Default:** + +``` +{"cert":"default","requireClientAuth":false} +``` + +### [listeners.admin.tls.cert](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.tls.cert) + +Name of the Certificate used for TLS (must match a Certificate name that is registered in tls.certs). + +**Default:** `"default"` + +### [listeners.admin.tls.requireClientAuth](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.admin.tls.requireClientAuth) + +If true, the truststore file for this listener is included in the ConfigMap. + +**Default:** `false` + +### [listeners.http](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.http) + +HTTP API listeners (aka PandaProxy). + +**Default:** + +``` +{"authenticationMethod":null,"enabled":true,"external":{"default":{"advertisedPorts":[30082],"authenticationMethod":null,"port":8083,"tls":{"cert":"external","requireClientAuth":false}}},"kafkaEndpoint":"default","port":8082,"tls":{"cert":"default","requireClientAuth":false}} +``` + +### [listeners.kafka](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.kafka) + +Kafka API listeners. + +**Default:** + +``` +{"authenticationMethod":null,"external":{"default":{"advertisedPorts":[31092],"authenticationMethod":null,"port":9094,"tls":{"cert":"external"}}},"port":9093,"tls":{"cert":"default","requireClientAuth":false}} +``` + +### [listeners.kafka.external.default.advertisedPorts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.kafka.external.default.advertisedPorts) + +If undefined, `listeners.kafka.external.default.port` is used. + +**Default:** `[31092]` + +### [listeners.kafka.external.default.port](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.kafka.external.default.port) + +The port used for external client connections. + +**Default:** `9094` + +### [listeners.kafka.port](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.kafka.port) + +The port for internal client connections. + +**Default:** `9093` + +### [listeners.rpc](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.rpc) + +RPC listener (this is never externally accessible). + +**Default:** + +``` +{"port":33145,"tls":{"cert":"default","requireClientAuth":false}} +``` + +### [listeners.schemaRegistry](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=listeners.schemaRegistry) + +Schema registry listeners. + +**Default:** + +``` +{"authenticationMethod":null,"enabled":true,"external":{"default":{"advertisedPorts":[30081],"authenticationMethod":null,"port":8084,"tls":{"cert":"external","requireClientAuth":false}}},"kafkaEndpoint":"default","port":8081,"tls":{"cert":"default","requireClientAuth":false}} +``` + +### [logging](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=logging) + +Log-level settings. + +**Default:** + +``` +{"logLevel":"info","usageStats":{"enabled":true}} +``` + +### [logging.logLevel](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=logging.logLevel) + +Log level Valid values (from least to most verbose) are: `warn`, `info`, `debug`, and `trace`. + +**Default:** `"info"` + +### [logging.usageStats](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=logging.usageStats) + +Send usage statistics back to Redpanda Data. For details, see the [stats reporting documentation](https://docs.redpanda.com/docs/cluster-administration/monitoring/#stats-reporting). + +**Default:** `{"enabled":true}` + +### [monitoring](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=monitoring) + +Monitoring. This will create a ServiceMonitor that can be used by Prometheus-Operator or VictoriaMetrics-Operator to scrape the metrics. + +**Default:** + +``` +{"enabled":false,"labels":{},"scrapeInterval":"30s"} +``` + +### [nameOverride](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=nameOverride) + +Override `redpanda.name` template. + +**Default:** `""` + +### [nodeSelector](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=nodeSelector) + +Node selection constraints for scheduling Pods, can override this for StatefulSets. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). + +**Default:** `{}` + +### [post_install_job.affinity](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=post_install_job.affinity) + +**Default:** `{}` + +### [post_install_job.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=post_install_job.enabled) + +**Default:** `true` + +### [post_install_job.podTemplate.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=post_install_job.podTemplate.annotations) + +Additional annotations to apply to the Pods of this Job. + +**Default:** `{}` + +### [post_install_job.podTemplate.labels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=post_install_job.podTemplate.labels) + +Additional labels to apply to the Pods of this Job. + +**Default:** `{}` + +### [post_install_job.podTemplate.spec](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=post_install_job.podTemplate.spec) + +A subset of Kubernetes' PodSpec type that will be merged into the final PodSpec. See [Merge Semantics](#merging-semantics) for details. + +**Default:** + +``` +{"containers":[{"env":[],"name":"post-install","securityContext":{}}],"securityContext":{}} +``` + +### [rackAwareness](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rackAwareness) + +Rack Awareness settings. For details, see the [Rack Awareness documentation](https://docs.redpanda.com/docs/manage/kubernetes/kubernetes-rack-awareness/). + +**Default:** + +``` +{"enabled":false,"nodeAnnotation":"topology.kubernetes.io/zone"} +``` + +### [rackAwareness.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rackAwareness.enabled) + +When running in multiple racks or availability zones, use a Kubernetes Node annotation value as the Redpanda rack value. Enabling this requires running with a service account with "get" Node permissions. To have the Helm chart configure these permissions, set `serviceAccount.create=true` and `rbac.enabled=true`. + +**Default:** `false` + +### [rackAwareness.nodeAnnotation](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rackAwareness.nodeAnnotation) + +The common well-known annotation to use as the rack ID. Override this only if you use a custom Node annotation. + +**Default:** + +``` +"topology.kubernetes.io/zone" +``` + +### [rbac](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rbac) + +Role Based Access Control. + +**Default:** + +``` +{"annotations":{},"enabled":false} +``` + +### [rbac.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rbac.annotations) + +Annotations to add to the `rbac` resources. + +**Default:** `{}` + +### [rbac.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=rbac.enabled) + +Enable for features that need extra privileges. If you use the Redpanda Operator, you must deploy it with the `--set rbac.createRPKBundleCRs=true` flag to give it the required ClusterRoles. + +**Default:** `false` + +### [resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources) + +Pod resource management. This section simplifies resource allocation by providing a single location where resources are defined. Helm sets these resource values within the `statefulset.yaml` and `configmap.yaml` templates. The default values are for a development environment. Production-level values and other considerations are documented, where those values are different from the default. For details, see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/). + +**Default:** + +``` +{"cpu":{"cores":1},"memory":{"container":{"max":"2.5Gi"}}} +``` + +### [resources.cpu](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources.cpu) + +CPU resources. For details, see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/#configure-cpu-resources). + +**Default:** `{"cores":1}` + +### [resources.cpu.cores](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources.cpu.cores) + +Redpanda makes use of a thread per core model. For details, see this [blog](https://redpanda.com/blog/tpc-buffers). For this reason, Redpanda should only be given full cores. Note: You can increase cores, but decreasing cores is not currently supported. See the [GitHub issue](https://github.com/redpanda-data/redpanda/issues/350). This setting is equivalent to `--smp`, `resources.requests.cpu`, and `resources.limits.cpu`. For production, use `4` or greater. To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for CPU resource requests and limits. This policy gives the Pods running Redpanda brokers access to exclusive CPUs on the node. See https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy. + +**Default:** `1` + +### [resources.memory](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources.memory) + +Memory resources For details, see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/#configure-memory-resources). + +**Default:** + +``` +{"container":{"max":"2.5Gi"}} +``` + +### [resources.memory.container](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources.memory.container) + +Enables memory locking. For production, set to `true`. enable_memory_locking: false It is recommended to have at least 2Gi of memory per core for the Redpanda binary. This memory is taken from the total memory given to each container. The Helm chart allocates 80% of the container's memory to Redpanda, leaving the rest for the Seastar subsystem (reserveMemory) and other container processes. So at least 2.5Gi per core is recommended in order to ensure Redpanda has a full 2Gi. These values affect `--memory` and `--reserve-memory` flags passed to Redpanda and the memory requests/limits in the StatefulSet. Valid suffixes: k, M, G, T, P, Ki, Mi, Gi, Ti, Pi To create `Guaranteed` Pod QoS for Redpanda brokers, provide both container max and min values for the container. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a memory limit and a memory request. * For every container in the Pod, the memory limit must equal the memory request. + +**Default:** `{"max":"2.5Gi"}` + +### [resources.memory.container.max](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=resources.memory.container.max) + +Maximum memory count for each Redpanda broker. Equivalent to `resources.limits.memory`. For production, use `10Gi` or greater. + +**Default:** `"2.5Gi"` + +### [serviceAccount](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=serviceAccount) + +Service account management. + +**Default:** + +``` +{"annotations":{},"automountServiceAccountToken":false,"create":false,"name":""} +``` + +### [serviceAccount.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=serviceAccount.annotations) + +Annotations to add to the service account. + +**Default:** `{}` + +### [serviceAccount.automountServiceAccountToken](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=serviceAccount.automountServiceAccountToken) + +Specifies whether a service account should automount API-Credentials. The token is used in sidecars.controllers + +**Default:** `false` + +### [serviceAccount.create](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=serviceAccount.create) + +Specifies whether a service account should be created. + +**Default:** `false` + +### [serviceAccount.name](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=serviceAccount.name) + +The name of the service account to use. If not set and `serviceAccount.create` is `true`, a name is generated using the `redpanda.fullname` template. + +**Default:** `""` + +### [statefulset.additionalRedpandaCmdFlags](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.additionalRedpandaCmdFlags) + +Additional flags to pass to redpanda, + +**Default:** `[]` + +### [statefulset.additionalSelectorLabels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.additionalSelectorLabels) + +Additional labels to be added to statefulset label selector. For example, `my.k8s.service: redpanda`. + +**Default:** `{}` + +### [statefulset.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.annotations) + +DEPRECATED Please use statefulset.podTemplate.annotations. Annotations are used only for `Statefulset.spec.template.metadata.annotations`. The StatefulSet does not have any dedicated annotation. + +**Default:** `{}` + +### [statefulset.budget.maxUnavailable](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.budget.maxUnavailable) + +**Default:** `1` + +### [statefulset.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.extraVolumes](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.extraVolumes) + +**Default:** `""` + +### [statefulset.initContainerImage.repository](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainerImage.repository) + +**Default:** `"busybox"` + +### [statefulset.initContainerImage.tag](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainerImage.tag) + +**Default:** `"latest"` + +### [statefulset.initContainers.configurator.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.configurator.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.initContainers.configurator.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.configurator.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. + +**Default:** `{}` + +### [statefulset.initContainers.extraInitContainers](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.extraInitContainers) + +**Default:** `""` + +### [statefulset.initContainers.fsValidator.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.fsValidator.enabled) + +**Default:** `false` + +### [statefulset.initContainers.fsValidator.expectedFS](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.fsValidator.expectedFS) + +**Default:** `"xfs"` + +### [statefulset.initContainers.fsValidator.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.fsValidator.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.initContainers.fsValidator.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.fsValidator.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. + +**Default:** `{}` + +### [statefulset.initContainers.setDataDirOwnership.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.setDataDirOwnership.enabled) + +In environments where root is not allowed, you cannot change the ownership of files and directories. Enable `setDataDirOwnership` when using default minikube cluster configuration. + +**Default:** `false` + +### [statefulset.initContainers.setDataDirOwnership.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.setDataDirOwnership.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.initContainers.setDataDirOwnership.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.setDataDirOwnership.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. + +**Default:** `{}` + +### [statefulset.initContainers.setTieredStorageCacheDirOwnership.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.setTieredStorageCacheDirOwnership.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.initContainers.setTieredStorageCacheDirOwnership.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.setTieredStorageCacheDirOwnership.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. + +**Default:** `{}` + +### [statefulset.initContainers.tuning.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.tuning.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.initContainers.tuning.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.initContainers.tuning.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. + +**Default:** `{}` + +### [statefulset.livenessProbe.failureThreshold](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.livenessProbe.failureThreshold) + +**Default:** `3` + +### [statefulset.livenessProbe.initialDelaySeconds](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.livenessProbe.initialDelaySeconds) + +**Default:** `10` + +### [statefulset.livenessProbe.periodSeconds](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.livenessProbe.periodSeconds) + +**Default:** `10` + +### [statefulset.nodeSelector](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.nodeSelector) + +Node selection constraints for scheduling Pods of this StatefulSet. These constraints override the global `nodeSelector` value. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). + +**Default:** `{}` + +### [statefulset.podAffinity](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAffinity) + +Inter-Pod Affinity rules for scheduling Pods of this StatefulSet. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + +**Default:** `{}` + +### [statefulset.podAntiAffinity](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAntiAffinity) + +Anti-affinity rules for scheduling Pods of this StatefulSet. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). You may either edit the default settings for anti-affinity rules, or specify new anti-affinity rules to use instead of the defaults. + +**Default:** + +``` +{"custom":{},"topologyKey":"kubernetes.io/hostname","type":"hard","weight":100} +``` + +### [statefulset.podAntiAffinity.custom](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAntiAffinity.custom) + +Change `podAntiAffinity.type` to `custom` and provide your own podAntiAffinity rules here. + +**Default:** `{}` + +### [statefulset.podAntiAffinity.topologyKey](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAntiAffinity.topologyKey) + +The topologyKey to be used. Can be used to spread across different nodes, AZs, regions etc. + +**Default:** `"kubernetes.io/hostname"` + +### [statefulset.podAntiAffinity.type](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAntiAffinity.type) + +Valid anti-affinity types are `soft`, `hard`, or `custom`. Use `custom` if you want to supply your own anti-affinity rules in the `podAntiAffinity.custom` object. + +**Default:** `"hard"` + +### [statefulset.podAntiAffinity.weight](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podAntiAffinity.weight) + +Weight for `soft` anti-affinity rules. Does not apply to other anti-affinity types. + +**Default:** `100` + +### [statefulset.podTemplate.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podTemplate.annotations) + +Additional annotations to apply to the Pods of the StatefulSet. + +**Default:** `{}` + +### [statefulset.podTemplate.labels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podTemplate.labels) + +Additional labels to apply to the Pods of the StatefulSet. + +**Default:** `{}` + +### [statefulset.podTemplate.spec](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.podTemplate.spec) + +A subset of Kubernetes' PodSpec type that will be merged into the final PodSpec. See [Merge Semantics](#merging-semantics) for details. + +**Default:** + +``` +{"containers":[{"env":[],"name":"redpanda","securityContext":{}}],"securityContext":{}} +``` + +### [statefulset.priorityClassName](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.priorityClassName) + +PriorityClassName given to Pods of this StatefulSet. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). + +**Default:** `""` + +### [statefulset.readinessProbe.failureThreshold](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.readinessProbe.failureThreshold) + +**Default:** `3` + +### [statefulset.readinessProbe.initialDelaySeconds](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.readinessProbe.initialDelaySeconds) + +**Default:** `1` + +### [statefulset.readinessProbe.periodSeconds](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.readinessProbe.periodSeconds) + +**Default:** `10` + +### [statefulset.readinessProbe.successThreshold](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.readinessProbe.successThreshold) + +**Default:** `1` + +### [statefulset.replicas](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.replicas) + +Number of Redpanda brokers (Redpanda Data recommends setting this to the number of worker nodes in the cluster) + +**Default:** `3` + +### [statefulset.securityContext](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.securityContext) + +DEPRECATED: Prefer to use podTemplate.spec.securityContext or podTemplate.spec.containers[0].securityContext. + +**Default:** + +``` +{"fsGroup":101,"fsGroupChangePolicy":"OnRootMismatch","runAsUser":101} +``` + +### [statefulset.sideCars.configWatcher.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.configWatcher.enabled) + +**Default:** `true` + +### [statefulset.sideCars.configWatcher.extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.configWatcher.extraVolumeMounts) + +**Default:** `""` + +### [statefulset.sideCars.configWatcher.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.configWatcher.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a memory limit and a memory request. * For every container in the Pod, the memory limit must equal the memory request. * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for CPU resource requests and limits. This policy gives the Pods running Redpanda brokers access to exclusive CPUs on the node. For details, see https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy + +**Default:** `{}` + +### [statefulset.sideCars.configWatcher.securityContext](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.configWatcher.securityContext) + +**Default:** `{}` + +### [statefulset.sideCars.controllers.createRBAC](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.createRBAC) + +**Default:** `true` + +### [statefulset.sideCars.controllers.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.enabled) + +**Default:** `false` + +### [statefulset.sideCars.controllers.healthProbeAddress](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.healthProbeAddress) + +**Default:** `":8085"` + +### [statefulset.sideCars.controllers.image.repository](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.image.repository) + +**Default:** + +``` +"docker.redpanda.com/redpandadata/redpanda-operator" +``` + +### [statefulset.sideCars.controllers.image.tag](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.image.tag) + +**Default:** `"v2.2.5-24.2.7"` + +### [statefulset.sideCars.controllers.metricsAddress](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.metricsAddress) + +**Default:** `":9082"` + +### [statefulset.sideCars.controllers.resources](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.resources) + +To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. * Every container in the Pod must have a CPU limit and a CPU request. * For every container in the Pod, the CPU limit must equal the CPU request. To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for CPU resource requests and limits. This policy gives the Pods running Redpanda brokers access to exclusive CPUs on the node. For details, see https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy + +**Default:** `{}` + +### [statefulset.sideCars.controllers.run[0]](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.run[0]) + +**Default:** `"all"` + +### [statefulset.sideCars.controllers.securityContext](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.sideCars.controllers.securityContext) + +**Default:** `{}` + +### [statefulset.startupProbe](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.startupProbe) + +Adjust the period for your probes to meet your needs. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). + +**Default:** + +``` +{"failureThreshold":120,"initialDelaySeconds":1,"periodSeconds":10} +``` + +### [statefulset.terminationGracePeriodSeconds](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.terminationGracePeriodSeconds) + +Termination grace period in seconds is time required to execute preStop hook which puts particular Redpanda Pod (process/container) into maintenance mode. Before settle down on particular value please put Redpanda under load and perform rolling upgrade or rolling restart. That value needs to accommodate two processes: * preStop hook needs to put Redpanda into maintenance mode * after preStop hook Redpanda needs to handle gracefully SIGTERM signal Both processes are executed sequentially where preStop hook has hard deadline in the middle of terminationGracePeriodSeconds. REF: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#hook-handler-execution https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#pod-termination + +**Default:** `90` + +### [statefulset.tolerations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.tolerations) + +Taints to be tolerated by Pods of this StatefulSet. These tolerations override the global tolerations value. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + +**Default:** `[]` + +### [statefulset.topologySpreadConstraints[0].maxSkew](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.topologySpreadConstraints[0].maxSkew) + +**Default:** `1` + +### [statefulset.topologySpreadConstraints[0].topologyKey](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.topologySpreadConstraints[0].topologyKey) + +**Default:** + +``` +"topology.kubernetes.io/zone" +``` + +### [statefulset.topologySpreadConstraints[0].whenUnsatisfiable](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.topologySpreadConstraints[0].whenUnsatisfiable) + +**Default:** `"ScheduleAnyway"` + +### [statefulset.updateStrategy.type](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=statefulset.updateStrategy.type) + +**Default:** `"RollingUpdate"` + +### [storage](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage) + +Persistence settings. For details, see the [storage documentation](https://docs.redpanda.com/docs/manage/kubernetes/configure-storage/). + +**Default:** + +``` +{"hostPath":"","persistentVolume":{"annotations":{},"enabled":true,"labels":{},"nameOverwrite":"","size":"20Gi","storageClass":""},"tiered":{"config":{"cloud_storage_cache_size":5368709120,"cloud_storage_enable_remote_read":true,"cloud_storage_enable_remote_write":true,"cloud_storage_enabled":false},"credentialsSecretRef":{"accessKey":{"configurationKey":"cloud_storage_access_key"},"secretKey":{"configurationKey":"cloud_storage_secret_key"}},"hostPath":"","mountType":"emptyDir","persistentVolume":{"annotations":{},"labels":{},"storageClass":""}}} +``` + +### [storage.hostPath](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.hostPath) + +Absolute path on the host to store Redpanda's data. If unspecified, then an `emptyDir` volume is used. If specified but `persistentVolume.enabled` is true, `storage.hostPath` has no effect. + +**Default:** `""` + +### [storage.persistentVolume](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.persistentVolume) + +If `persistentVolume.enabled` is true, a PersistentVolumeClaim is created and used to store Redpanda's data. Otherwise, `storage.hostPath` is used. + +**Default:** + +``` +{"annotations":{},"enabled":true,"labels":{},"nameOverwrite":"","size":"20Gi","storageClass":""} +``` + +### [storage.persistentVolume.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.persistentVolume.annotations) + +Additional annotations to apply to the created PersistentVolumeClaims. + +**Default:** `{}` + +### [storage.persistentVolume.labels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.persistentVolume.labels) + +Additional labels to apply to the created PersistentVolumeClaims. + +**Default:** `{}` + +### [storage.persistentVolume.nameOverwrite](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.persistentVolume.nameOverwrite) + +Option to change volume claim template name for tiered storage persistent volume if tiered.mountType is set to `persistentVolume` + +**Default:** `""` + +### [storage.persistentVolume.storageClass](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.persistentVolume.storageClass) + +To disable dynamic provisioning, set to `-`. If undefined or empty (default), then no storageClassName spec is set, and the default dynamic provisioner is chosen (gp2 on AWS, standard on GKE, AWS & OpenStack). + +**Default:** `""` + +### [storage.tiered.config](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.config) + +Tiered Storage settings Requires `enterprise.licenseKey` or `enterprised.licenseSecretRef` For details, see the [Tiered Storage documentation](https://docs.redpanda.com/docs/manage/kubernetes/tiered-storage/). For a list of properties, see [Object Storage Properties](https://docs.redpanda.com/current/reference/properties/object-storage-properties/). + +**Default:** + +``` +{"cloud_storage_cache_size":5368709120,"cloud_storage_enable_remote_read":true,"cloud_storage_enable_remote_write":true,"cloud_storage_enabled":false} +``` + +### [storage.tiered.config.cloud_storage_cache_size](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.config.cloud_storage_cache_size) + +Maximum size of the disk cache used by Tiered Storage. Default is 20 GiB. See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_cache_size). + +**Default:** `5368709120` + +### [storage.tiered.config.cloud_storage_enable_remote_read](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.config.cloud_storage_enable_remote_read) + +Cluster level default remote read configuration for new topics. See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enable_remote_read). + +**Default:** `true` + +### [storage.tiered.config.cloud_storage_enable_remote_write](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.config.cloud_storage_enable_remote_write) + +Cluster level default remote write configuration for new topics. See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enable_remote_write). + +**Default:** `true` + +### [storage.tiered.config.cloud_storage_enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.config.cloud_storage_enabled) + +Global flag that enables Tiered Storage if a license key is provided. See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enabled). + +**Default:** `false` + +### [storage.tiered.hostPath](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.hostPath) + +Absolute path on the host to store Redpanda's Tiered Storage cache. + +**Default:** `""` + +### [storage.tiered.persistentVolume.annotations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.persistentVolume.annotations) + +Additional annotations to apply to the created PersistentVolumeClaims. + +**Default:** `{}` + +### [storage.tiered.persistentVolume.labels](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.persistentVolume.labels) + +Additional labels to apply to the created PersistentVolumeClaims. + +**Default:** `{}` + +### [storage.tiered.persistentVolume.storageClass](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=storage.tiered.persistentVolume.storageClass) + +To disable dynamic provisioning, set to "-". If undefined or empty (default), then no storageClassName spec is set, and the default dynamic provisioner is chosen (gp2 on AWS, standard on GKE, AWS & OpenStack). + +**Default:** `""` + +### [tests.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tests.enabled) + +**Default:** `true` + +### [tls](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls) + +TLS settings. For details, see the [TLS documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/kubernetes-tls/). + +**Default:** + +``` +{"certs":{"default":{"caEnabled":true},"external":{"caEnabled":true}},"enabled":true} +``` + +### [tls.certs](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.certs) + +List all Certificates here, then you can reference a specific Certificate's name in each listener's `listeners..tls.cert` setting. + +**Default:** + +``` +{"default":{"caEnabled":true},"external":{"caEnabled":true}} +``` + +### [tls.certs.default](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.certs.default) + +This key is the Certificate name. To apply the Certificate to a specific listener, reference the Certificate's name in `listeners..tls.cert`. + +**Default:** `{"caEnabled":true}` + +### [tls.certs.default.caEnabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.certs.default.caEnabled) + +Indicates whether or not the Secret holding this certificate includes a `ca.crt` key. When `true`, chart managed clients, such as rpk, will use `ca.crt` for certificate verification and listeners with `require_client_auth` and no explicit `truststore` will use `ca.crt` as their `truststore_file` for verification of client certificates. When `false`, chart managed clients will use `tls.crt` for certificate verification and listeners with `require_client_auth` and no explicit `truststore` will use the container's CA certificates. + +**Default:** `true` + +### [tls.certs.external](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.certs.external) + +Example external tls configuration uncomment and set the right key to the listeners that require them also enable the tls setting for those listeners. + +**Default:** `{"caEnabled":true}` + +### [tls.certs.external.caEnabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.certs.external.caEnabled) + +Indicates whether or not the Secret holding this certificate includes a `ca.crt` key. When `true`, chart managed clients, such as rpk, will use `ca.crt` for certificate verification and listeners with `require_client_auth` and no explicit `truststore` will use `ca.crt` as their `truststore_file` for verification of client certificates. When `false`, chart managed clients will use `tls.crt` for certificate verification and listeners with `require_client_auth` and no explicit `truststore` will use the container's CA certificates. + +**Default:** `true` + +### [tls.enabled](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tls.enabled) + +Enable TLS globally for all listeners. Each listener must include a Certificate name in its `.tls` object. To allow you to enable TLS for individual listeners, Certificates in `auth.tls.certs` are always loaded, even if `tls.enabled` is `false`. See `listeners..tls.enabled`. + +**Default:** `true` + +### [tolerations](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tolerations) + +Taints to be tolerated by Pods, can override this for StatefulSets. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + +**Default:** `[]` + +### [tuning](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tuning) + +Redpanda tuning settings. Each is set to their default values in Redpanda. + +**Default:** `{"tune_aio_events":true}` + +### [tuning.tune_aio_events](https://artifacthub.io/packages/helm/redpanda-data/redpanda?modal=values&path=tuning.tune_aio_events) + +Increase the maximum number of outstanding asynchronous IO operations if the current value is below a certain threshold. This allows Redpanda to make as many simultaneous IO requests as possible, increasing throughput. When this option is enabled, Helm creates a privileged container. If your security profile does not allow this, you can disable this container by setting `tune_aio_events` to `false`. For more details, see the [tuning documentation](https://docs.redpanda.com/docs/deploy/deployment-option/self-hosted/kubernetes/kubernetes-tune-workers/). + +**Default:** `true` + +## Merging Semantics + +The redpanda chart implements a form of object merging that's roughly a +middleground of [JSON Merge Patch][k8s.jsonmp] and [Kubernetes' Strategic Merge +Patch][k8s.smp]. This is done to aid end users in setting or overriding fields +that are not directly exposed via the chart. + +- Directives are not supported. +- List fields that are merged by a unique key in Kubernetes' SMP (e.g. + `containers`, `env`) will be merged in a similar awy. +- Only fields explicitly allowed by the chart's JSON schema will be merged. +- Additional containers that are not present in the original value will NOT be added. + +[k8s.smp]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/update-api-object-kubectl-patch/#use-a-strategic-merge-patch-to-update-a-deployment +[k8s.jsonmp]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/update-api-object-kubectl-patch/#use-a-json-merge-patch-to-update-a-deployment diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/.helmignore b/charts/redpanda/redpanda/5.9.10/charts/connectors/.helmignore new file mode 100644 index 0000000000..2e271ea0fc --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/.helmignore @@ -0,0 +1,29 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +README.md.gotmpl +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +*.go +testdata/ +ci/ +examples/ \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/Chart.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/Chart.yaml new file mode 100644 index 0000000000..0dd2396e50 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/Chart.yaml @@ -0,0 +1,25 @@ +annotations: + artifacthub.io/images: | + - name: connectors + image: docker.redpanda.com/redpandadata/connectors:v1.0.31 + - name: rpk + image: docker.redpanda.com/redpandadata/redpanda:latest + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Documentation + url: https://docs.redpanda.com + - name: "Helm (>= 3.6.0)" + url: https://helm.sh/docs/intro/install/ +apiVersion: v2 +appVersion: v1.0.31 +description: Redpanda managed Connectors helm chart +icon: https://images.ctfassets.net/paqvtpyf8rwu/3cYHw5UzhXCbKuR24GDFGO/73fb682e6157d11c10d5b2b5da1d5af0/skate-stand-panda.svg +kubeVersion: ^1.21.0-0 +maintainers: +- name: redpanda-data + url: https://github.com/orgs/redpanda-data/people +name: connectors +sources: +- https://github.com/redpanda-data/helm-charts +type: application +version: 0.1.13 diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/LICENSE b/charts/redpanda/redpanda/5.9.10/charts/connectors/LICENSE new file mode 100644 index 0000000000..261eeb9e9f --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/README.md b/charts/redpanda/redpanda/5.9.10/charts/connectors/README.md new file mode 100644 index 0000000000..2cb1438568 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/README.md @@ -0,0 +1,574 @@ +# Redpanda Connectors Helm Chart Specification +--- +description: Find the default values and descriptions of settings in the Redpanda Connectors Helm chart. +--- + +![Version: 0.1.13](https://img.shields.io/badge/Version-0.1.13-informational?style=flat-square) ![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![AppVersion: v1.0.31](https://img.shields.io/badge/AppVersion-v1.0.31-informational?style=flat-square) + +This page describes the official Redpanda Connectors Helm Chart. In particular, this page describes the contents of the chart’s [`values.yaml` file](https://github.com/redpanda-data/helm-charts/blob/main/charts/connectors/values.yaml). Each of the settings is listed and described on this page, along with any default values. + +For instructions on how to install and use the chart, including how to override and customize the chart’s values, refer to the [deployment documentation](https://docs.redpanda.com/current/deploy/deployment-option/self-hosted/kubernetes/k-deploy-connectors/). + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.13.1](https://github.com/norwoodj/helm-docs/releases/v1.13.1) + +## Source Code + +* + +## Requirements + +Kubernetes: `^1.21.0-0` + +## Settings + +### [auth](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=auth) + +Authentication settings. For details, see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/). The first line of the secret file is used. So the first superuser is used to authenticate to the Redpanda cluster. + +**Default:** + +``` +{"sasl":{"enabled":false,"mechanism":"scram-sha-512","secretRef":"","userName":""}} +``` + +### [auth.sasl.mechanism](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=auth.sasl.mechanism) + +The authentication mechanism to use for the superuser. Options are `scram-sha-256` and `scram-sha-512`. + +**Default:** `"scram-sha-512"` + +### [auth.sasl.secretRef](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=auth.sasl.secretRef) + +A Secret that contains your SASL user password. + +**Default:** `""` + +### [commonLabels](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=commonLabels) + +Additional labels to add to all Kubernetes objects. For example, `my.k8s.service: redpanda`. + +**Default:** `{}` + +### [connectors.additionalConfiguration](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.additionalConfiguration) + +A placeholder for any Java configuration settings for Kafka Connect that are not explicitly defined in this Helm chart. Java configuration settings are passed to the Kafka Connect startup script. + +**Default:** `""` + +### [connectors.bootstrapServers](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.bootstrapServers) + +A comma-separated list of Redpanda broker addresses in the format of IP:Port or DNS:Port. Kafka Connect uses this to connect to the Redpanda/Kafka cluster. + +**Default:** `""` + +### [connectors.brokerTLS.ca.secretNameOverwrite](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.ca.secretNameOverwrite) + +If `secretRef` points to a Secret where the certificate authority (CA) is not under the `ca.crt` key, use `secretNameOverwrite` to overwrite it e.g. `corp-ca.crt`. + +**Default:** `""` + +### [connectors.brokerTLS.ca.secretRef](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.ca.secretRef) + +The name of the Secret where the ca.crt file content is located. + +**Default:** `""` + +### [connectors.brokerTLS.cert.secretNameOverwrite](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.cert.secretNameOverwrite) + +If secretRef points to secret where client signed certificate is not under tls.crt key then please use secretNameOverwrite to overwrite it e.g. corp-tls.crt + +**Default:** `""` + +### [connectors.brokerTLS.cert.secretRef](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.cert.secretRef) + +The name of the secret where client signed certificate is located + +**Default:** `""` + +### [connectors.brokerTLS.enabled](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.enabled) + +**Default:** `false` + +### [connectors.brokerTLS.key.secretNameOverwrite](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.key.secretNameOverwrite) + +If secretRef points to secret where client private key is not under tls.key key then please use secretNameOverwrite to overwrite it e.g. corp-tls.key + +**Default:** `""` + +### [connectors.brokerTLS.key.secretRef](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.brokerTLS.key.secretRef) + +The name of the secret where client private key is located + +**Default:** `""` + +### [connectors.groupID](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.groupID) + +A unique string that identifies the Kafka Connect cluster. It's used in the formation of the internal topic names, ensuring that multiple Kafka Connect clusters can connect to the same Redpanda cluster without interfering with each other. + +**Default:** `"connectors-cluster"` + +### [connectors.producerBatchSize](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.producerBatchSize) + +The number of bytes of records a producer will attempt to batch together before sending to Redpanda. Batching improves throughput. + +**Default:** `131072` + +### [connectors.producerLingerMS](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.producerLingerMS) + +The time, in milliseconds, that a producer will wait before sending a batch of records. Waiting allows the producer to gather more records in the same batch and improve throughput. + +**Default:** `1` + +### [connectors.restPort](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.restPort) + +The port on which the Kafka Connect REST API listens. The API is used for administrative tasks. + +**Default:** `8083` + +### [connectors.schemaRegistryURL](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.schemaRegistryURL) + +**Default:** `""` + +### [connectors.secretManager.connectorsPrefix](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.secretManager.connectorsPrefix) + +**Default:** `""` + +### [connectors.secretManager.consolePrefix](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.secretManager.consolePrefix) + +**Default:** `""` + +### [connectors.secretManager.enabled](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.secretManager.enabled) + +**Default:** `false` + +### [connectors.secretManager.region](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.secretManager.region) + +**Default:** `""` + +### [connectors.storage.remote](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.remote) + +Indicates if read and write operations for the respective topics are allowed remotely. + +**Default:** + +``` +{"read":{"config":false,"offset":false,"status":false},"write":{"config":false,"offset":false,"status":false}} +``` + +### [connectors.storage.replicationFactor](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.replicationFactor) + +The number of replicas for each of the internal topics that Kafka Connect uses. + +**Default:** + +``` +{"config":-1,"offset":-1,"status":-1} +``` + +### [connectors.storage.replicationFactor.config](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.replicationFactor.config) + +Replication factor for the configuration topic. + +**Default:** `-1` + +### [connectors.storage.replicationFactor.offset](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.replicationFactor.offset) + +Replication factor for the offset topic. + +**Default:** `-1` + +### [connectors.storage.replicationFactor.status](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.replicationFactor.status) + +Replication factor for the status topic. + +**Default:** `-1` + +### [connectors.storage.topic.config](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.topic.config) + +The name of the internal topic that Kafka Connect uses to store connector and task configurations. + +**Default:** + +``` +"_internal_connectors_configs" +``` + +### [connectors.storage.topic.offset](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.topic.offset) + +The name of the internal topic that Kafka Connect uses to store source connector offsets. + +**Default:** + +``` +"_internal_connectors_offsets" +``` + +### [connectors.storage.topic.status](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=connectors.storage.topic.status) + +The name of the internal topic that Kafka Connect uses to store connector and task status updates. + +**Default:** + +``` +"_internal_connectors_status" +``` + +### [container.javaGCLogEnabled](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=container.javaGCLogEnabled) + +**Default:** `"false"` + +### [container.resources](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=container.resources) + +Pod resource management. + +**Default:** + +``` +{"javaMaxHeapSize":"2G","limits":{"cpu":"1","memory":"2350Mi"},"request":{"cpu":"1","memory":"2350Mi"}} +``` + +### [container.resources.javaMaxHeapSize](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=container.resources.javaMaxHeapSize) + +Java maximum heap size must not be greater than `container.resources.limits.memory`. + +**Default:** `"2G"` + +### [container.securityContext](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=container.securityContext) + +Security context for the Redpanda Connectors container. See also `deployment.securityContext` for Pod-level settings. + +**Default:** + +``` +{"allowPrivilegeEscalation":false} +``` + +### [deployment.annotations](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.annotations) + +Additional annotations to apply to the Pods of this Deployment. + +**Default:** `{}` + +### [deployment.budget.maxUnavailable](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.budget.maxUnavailable) + +**Default:** `1` + +### [deployment.create](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.create) + +**Default:** `true` + +### [deployment.extraEnv](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.extraEnv) + +Additional environment variables for the Pods. + +**Default:** `[]` + +### [deployment.extraEnvFrom](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.extraEnvFrom) + +Configure extra environment variables from Secrets and ConfigMaps. + +**Default:** `[]` + +### [deployment.livenessProbe](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.livenessProbe) + +Adjust the period for your probes to meet your needs. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). + +**Default:** + +``` +{"failureThreshold":3,"initialDelaySeconds":10,"periodSeconds":10,"successThreshold":1,"timeoutSeconds":1} +``` + +### [deployment.nodeAffinity](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.nodeAffinity) + +Node Affinity rules for scheduling Pods of this Deployment. The suggestion would be to spread Pods according to topology zone. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity). + +**Default:** `{}` + +### [deployment.nodeSelector](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.nodeSelector) + +Node selection constraints for scheduling Pods of this Deployment. These constraints override the global `nodeSelector` value. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). + +**Default:** `{}` + +### [deployment.podAffinity](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAffinity) + +Inter-Pod Affinity rules for scheduling Pods of this Deployment. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + +**Default:** `{}` + +### [deployment.podAntiAffinity](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAntiAffinity) + +Anti-affinity rules for scheduling Pods of this Deployment. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). You may either edit the default settings for anti-affinity rules, or specify new anti-affinity rules to use instead of the defaults. + +**Default:** + +``` +{"custom":{},"topologyKey":"kubernetes.io/hostname","type":"hard","weight":100} +``` + +### [deployment.podAntiAffinity.custom](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAntiAffinity.custom) + +Change `podAntiAffinity.type` to `custom` and provide your own podAntiAffinity rules here. + +**Default:** `{}` + +### [deployment.podAntiAffinity.topologyKey](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAntiAffinity.topologyKey) + +The `topologyKey` to be used. Can be used to spread across different nodes, AZs, regions etc. + +**Default:** `"kubernetes.io/hostname"` + +### [deployment.podAntiAffinity.type](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAntiAffinity.type) + +Valid anti-affinity types are `soft`, `hard`, or `custom`. Use `custom` if you want to supply your own anti-affinity rules in the `podAntiAffinity.custom` object. + +**Default:** `"hard"` + +### [deployment.podAntiAffinity.weight](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.podAntiAffinity.weight) + +Weight for `soft` anti-affinity rules. Does not apply for other anti-affinity types. + +**Default:** `100` + +### [deployment.priorityClassName](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.priorityClassName) + +PriorityClassName given to Pods of this Deployment. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). + +**Default:** `""` + +### [deployment.progressDeadlineSeconds](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.progressDeadlineSeconds) + +The maximum time in seconds for a deployment to make progress before it is considered to be failed. The deployment controller will continue to process failed deployments and a condition with a ProgressDeadlineExceeded reason will be surfaced in the deployment status. Note that progress will not be estimated during the time a deployment is paused. + +**Default:** `600` + +### [deployment.readinessProbe.failureThreshold](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.readinessProbe.failureThreshold) + +**Default:** `2` + +### [deployment.readinessProbe.initialDelaySeconds](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.readinessProbe.initialDelaySeconds) + +**Default:** `60` + +### [deployment.readinessProbe.periodSeconds](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.readinessProbe.periodSeconds) + +**Default:** `10` + +### [deployment.readinessProbe.successThreshold](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.readinessProbe.successThreshold) + +**Default:** `3` + +### [deployment.readinessProbe.timeoutSeconds](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.readinessProbe.timeoutSeconds) + +**Default:** `5` + +### [deployment.restartPolicy](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.restartPolicy) + +**Default:** `"Always"` + +### [deployment.revisionHistoryLimit](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.revisionHistoryLimit) + +The number of old ReplicaSets to retain to allow rollback. This is a pointer to distinguish between explicit zero and not specified. + +**Default:** `10` + +### [deployment.schedulerName](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.schedulerName) + +**Default:** `""` + +### [deployment.securityContext.fsGroup](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.securityContext.fsGroup) + +**Default:** `101` + +### [deployment.securityContext.fsGroupChangePolicy](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.securityContext.fsGroupChangePolicy) + +**Default:** `"OnRootMismatch"` + +### [deployment.securityContext.runAsUser](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.securityContext.runAsUser) + +**Default:** `101` + +### [deployment.strategy.type](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.strategy.type) + +**Default:** `"RollingUpdate"` + +### [deployment.terminationGracePeriodSeconds](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.terminationGracePeriodSeconds) + +**Default:** `30` + +### [deployment.tolerations](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.tolerations) + +Taints to be tolerated by Pods of this Deployment. These tolerations override the global tolerations value. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + +**Default:** `[]` + +### [deployment.topologySpreadConstraints[0].maxSkew](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.topologySpreadConstraints[0].maxSkew) + +**Default:** `1` + +### [deployment.topologySpreadConstraints[0].topologyKey](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.topologySpreadConstraints[0].topologyKey) + +**Default:** + +``` +"topology.kubernetes.io/zone" +``` + +### [deployment.topologySpreadConstraints[0].whenUnsatisfiable](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=deployment.topologySpreadConstraints[0].whenUnsatisfiable) + +**Default:** `"ScheduleAnyway"` + +### [fullnameOverride](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=fullnameOverride) + +Override `connectors.fullname` template. + +**Default:** `""` + +### [image](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=image) + +Redpanda Docker image settings. + +**Default:** + +``` +{"pullPolicy":"IfNotPresent","repository":"docker.redpanda.com/redpandadata/connectors","tag":""} +``` + +### [image.pullPolicy](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=image.pullPolicy) + +The imagePullPolicy. If `image.tag` is 'latest', the default is `Always`. + +**Default:** `"IfNotPresent"` + +### [image.repository](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=image.repository) + +Docker repository from which to pull the Redpanda Docker image. + +**Default:** + +``` +"docker.redpanda.com/redpandadata/connectors" +``` + +### [image.tag](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=image.tag) + +The Redpanda version. See DockerHub for: [All stable versions](https://hub.docker.com/r/redpandadata/redpanda/tags) and [all unstable versions](https://hub.docker.com/r/redpandadata/redpanda-unstable/tags). + +**Default:** `Chart.appVersion`. + +### [imagePullSecrets](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=imagePullSecrets) + +Pull secrets may be used to provide credentials to image repositories See https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + +**Default:** `[]` + +### [logging](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=logging) + +Log-level settings. + +**Default:** `{"level":"warn"}` + +### [logging.level](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=logging.level) + +Log level Valid values (from least to most verbose) are: `error`, `warn`, `info` and `debug`. + +**Default:** `"warn"` + +### [monitoring](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=monitoring) + +Monitoring. When set to `true`, the Helm chart creates a PodMonitor that can be used by Prometheus-Operator or VictoriaMetrics-Operator to scrape the metrics. + +**Default:** + +``` +{"annotations":{},"enabled":false,"labels":{},"namespaceSelector":{"any":true},"scrapeInterval":"30s"} +``` + +### [nameOverride](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=nameOverride) + +Override `connectors.name` template. + +**Default:** `""` + +### [service](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=service) + +Service management. + +**Default:** + +``` +{"annotations":{},"name":"","ports":[{"name":"prometheus","port":9404}]} +``` + +### [service.annotations](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=service.annotations) + +Annotations to add to the Service. + +**Default:** `{}` + +### [service.name](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=service.name) + +The name of the service to use. If not set, a name is generated using the `connectors.fullname` template. + +**Default:** `""` + +### [serviceAccount](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=serviceAccount) + +ServiceAccount management. + +**Default:** + +``` +{"annotations":{},"create":false,"name":""} +``` + +### [serviceAccount.annotations](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=serviceAccount.annotations) + +Annotations to add to the ServiceAccount. + +**Default:** `{}` + +### [serviceAccount.create](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=serviceAccount.create) + +Specifies whether a ServiceAccount should be created. + +**Default:** `false` + +### [serviceAccount.name](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=serviceAccount.name) + +The name of the ServiceAccount to use. If not set and `serviceAccount.create` is `true`, a name is generated using the `connectors.fullname` template. + +**Default:** `""` + +### [storage.volumeMounts[0].mountPath](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=storage.volumeMounts[0].mountPath) + +**Default:** `"/tmp"` + +### [storage.volumeMounts[0].name](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=storage.volumeMounts[0].name) + +**Default:** `"rp-connect-tmp"` + +### [storage.volume[0].emptyDir.medium](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=storage.volume[0].emptyDir.medium) + +**Default:** `"Memory"` + +### [storage.volume[0].emptyDir.sizeLimit](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=storage.volume[0].emptyDir.sizeLimit) + +**Default:** `"5Mi"` + +### [storage.volume[0].name](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=storage.volume[0].name) + +**Default:** `"rp-connect-tmp"` + +### [test.create](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=test.create) + +**Default:** `true` + +### [tolerations](https://artifacthub.io/packages/helm/redpanda-data/connectors?modal=values&path=tolerations) + +Taints to be tolerated by Pods. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + +**Default:** `[]` + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_deployment.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_deployment.go.tpl new file mode 100644 index 0000000000..f785c1ad92 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_deployment.go.tpl @@ -0,0 +1,136 @@ +{{- /* Generated from "deployment.go" */ -}} + +{{- define "connectors.Deployment" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.deployment.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $topologySpreadConstraints := (coalesce nil) -}} +{{- range $_, $spread := $values.deployment.topologySpreadConstraints -}} +{{- $topologySpreadConstraints = (concat (default (list ) $topologySpreadConstraints) (list (mustMergeOverwrite (dict "maxSkew" 0 "topologyKey" "" "whenUnsatisfiable" "" ) (dict "labelSelector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) "maxSkew" ($spread.maxSkew | int) "topologyKey" $spread.topologyKey "whenUnsatisfiable" $spread.whenUnsatisfiable )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $ports := (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "containerPort" ($values.connectors.restPort | int) "name" "rest-api" "protocol" "TCP" ))) -}} +{{- range $_, $port := $values.service.ports -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" $port.name "containerPort" ($port.port | int) "protocol" "TCP" )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $podAntiAffinity := (coalesce nil) -}} +{{- if (ne $values.deployment.podAntiAffinity (coalesce nil)) -}} +{{- if (eq $values.deployment.podAntiAffinity.type "hard") -}} +{{- $podAntiAffinity = (mustMergeOverwrite (dict ) (dict "requiredDuringSchedulingIgnoredDuringExecution" (list (mustMergeOverwrite (dict "topologyKey" "" ) (dict "topologyKey" $values.deployment.podAntiAffinity.topologyKey "namespaces" (list $dot.Release.Namespace) "labelSelector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) ))) )) -}} +{{- else -}}{{- if (eq $values.deployment.podAntiAffinity.type "soft") -}} +{{- $podAntiAffinity = (mustMergeOverwrite (dict ) (dict "preferredDuringSchedulingIgnoredDuringExecution" (list (mustMergeOverwrite (dict "weight" 0 "podAffinityTerm" (dict "topologyKey" "" ) ) (dict "weight" $values.deployment.podAntiAffinity.weight "podAffinityTerm" (mustMergeOverwrite (dict "topologyKey" "" ) (dict "topologyKey" $values.deployment.podAntiAffinity.topologyKey "namespaces" (list $dot.Release.Namespace) "labelSelector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) )) ))) )) -}} +{{- else -}}{{- if (eq $values.deployment.podAntiAffinity.type "custom") -}} +{{- $podAntiAffinity = $values.deployment.podAntiAffinity.custom -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "strategy" (dict ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "apps/v1" "kind" "Deployment" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "connectors.Fullname" (dict "a" (list $dot) ))) "r") "labels" (merge (dict ) (get (fromJson (include "connectors.FullLabels" (dict "a" (list $dot) ))) "r") $values.deployment.annotations) )) "spec" (mustMergeOverwrite (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "strategy" (dict ) ) (dict "replicas" $values.deployment.replicas "progressDeadlineSeconds" ($values.deployment.progressDeadlineSeconds | int) "revisionHistoryLimit" $values.deployment.revisionHistoryLimit "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) "strategy" $values.deployment.strategy "template" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "annotations" $values.deployment.annotations "labels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict "containers" (coalesce nil) ) (dict "terminationGracePeriodSeconds" $values.deployment.terminationGracePeriodSeconds "affinity" (mustMergeOverwrite (dict ) (dict "nodeAffinity" $values.deployment.nodeAffinity "podAffinity" $values.deployment.podAffinity "podAntiAffinity" $podAntiAffinity )) "serviceAccountName" (get (fromJson (include "connectors.ServiceAccountName" (dict "a" (list $dot) ))) "r") "containers" (list (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "connectors-cluster" "image" (printf "%s:%s" $values.image.repository (get (fromJson (include "connectors.Tag" (dict "a" (list $dot) ))) "r")) "imagePullPolicy" $values.image.pullPolicy "securityContext" $values.container.securityContext "command" $values.deployment.command "env" (get (fromJson (include "connectors.env" (dict "a" (list $values) ))) "r") "envFrom" $values.deployment.extraEnvFrom "livenessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "httpGet" (mustMergeOverwrite (dict "port" 0 ) (dict "path" "/" "port" "rest-api" "scheme" "HTTP" )) )) (dict "initialDelaySeconds" ($values.deployment.livenessProbe.initialDelaySeconds | int) "timeoutSeconds" ($values.deployment.livenessProbe.timeoutSeconds | int) "periodSeconds" ($values.deployment.livenessProbe.periodSeconds | int) "successThreshold" ($values.deployment.livenessProbe.successThreshold | int) "failureThreshold" ($values.deployment.livenessProbe.failureThreshold | int) )) "readinessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "httpGet" (mustMergeOverwrite (dict "port" 0 ) (dict "path" "/connectors" "port" "rest-api" "scheme" "HTTP" )) )) (dict "initialDelaySeconds" ($values.deployment.readinessProbe.initialDelaySeconds | int) "timeoutSeconds" ($values.deployment.readinessProbe.timeoutSeconds | int) "periodSeconds" ($values.deployment.readinessProbe.periodSeconds | int) "successThreshold" ($values.deployment.readinessProbe.successThreshold | int) "failureThreshold" ($values.deployment.readinessProbe.failureThreshold | int) )) "ports" $ports "resources" (mustMergeOverwrite (dict ) (dict "requests" $values.container.resources.request "limits" $values.container.resources.limits )) "terminationMessagePath" "/dev/termination-log" "terminationMessagePolicy" "File" "volumeMounts" (get (fromJson (include "connectors.volumeMountss" (dict "a" (list $values) ))) "r") ))) "dnsPolicy" "ClusterFirst" "restartPolicy" $values.deployment.restartPolicy "schedulerName" $values.deployment.schedulerName "nodeSelector" $values.deployment.nodeSelector "imagePullSecrets" $values.imagePullSecrets "securityContext" $values.deployment.securityContext "tolerations" $values.deployment.tolerations "topologySpreadConstraints" $topologySpreadConstraints "volumes" (get (fromJson (include "connectors.volumes" (dict "a" (list $values) ))) "r") )) )) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.env" -}} +{{- $values := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $env := (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_CONFIGURATION" "value" (get (fromJson (include "connectors.connectorConfiguration" (dict "a" (list $values) ))) "r") )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_ADDITIONAL_CONFIGURATION" "value" $values.connectors.additionalConfiguration )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_BOOTSTRAP_SERVERS" "value" $values.connectors.bootstrapServers ))) -}} +{{- if (not (empty $values.connectors.schemaRegistryURL)) -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "SCHEMA_REGISTRY_URL" "value" $values.connectors.schemaRegistryURL )))) -}} +{{- end -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_GC_LOG_ENABLED" "value" $values.container.javaGCLogEnabled )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_HEAP_OPTS" "value" (printf "-Xms256M -Xmx%s" $values.container.resources.javaMaxHeapSize) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_LOG_LEVEL" "value" $values.logging.level )))) -}} +{{- if (get (fromJson (include "connectors.Auth.SASLEnabled" (dict "a" (list $values.auth) ))) "r") -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_SASL_USERNAME" "value" $values.auth.sasl.userName )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_SASL_MECHANISM" "value" $values.auth.sasl.mechanism )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_SASL_PASSWORD_FILE" "value" "rc-credentials/password" )))) -}} +{{- end -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_TLS_ENABLED" "value" (printf "%v" $values.connectors.brokerTLS.enabled) )))) -}} +{{- if (not (empty $values.connectors.brokerTLS.ca.secretRef)) -}} +{{- $ca := (default "ca.crt" $values.connectors.brokerTLS.ca.secretNameOverwrite) -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_TRUSTED_CERTS" "value" (printf "ca/%s" $ca) )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.cert.secretRef)) -}} +{{- $cert := (default "tls.crt" $values.connectors.brokerTLS.cert.secretNameOverwrite) -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_TLS_AUTH_CERT" "value" (printf "cert/%s" $cert) )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.key.secretRef)) -}} +{{- $key := (default "tls.key" $values.connectors.brokerTLS.key.secretNameOverwrite) -}} +{{- $env = (concat (default (list ) $env) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONNECT_TLS_AUTH_KEY" "value" (printf "key/%s" $key) )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $env) (default (list ) $values.deployment.extraEnv))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.connectorConfiguration" -}} +{{- $values := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $lines := (list (printf "rest.advertised.port=%d" ($values.connectors.restPort | int)) (printf "rest.port=%d" ($values.connectors.restPort | int)) "key.converter=org.apache.kafka.connect.converters.ByteArrayConverter" "value.converter=org.apache.kafka.connect.converters.ByteArrayConverter" (printf "group.id=%s" $values.connectors.groupID) (printf "offset.storage.topic=%s" $values.connectors.storage.topic.offset) (printf "config.storage.topic=%s" $values.connectors.storage.topic.config) (printf "status.storage.topic=%s" $values.connectors.storage.topic.status) (printf "offset.storage.redpanda.remote.read=%t" $values.connectors.storage.remote.read.offset) (printf "offset.storage.redpanda.remote.write=%t" $values.connectors.storage.remote.write.offset) (printf "config.storage.redpanda.remote.read=%t" $values.connectors.storage.remote.read.config) (printf "config.storage.redpanda.remote.write=%t" $values.connectors.storage.remote.write.config) (printf "status.storage.redpanda.remote.read=%t" $values.connectors.storage.remote.read.status) (printf "status.storage.redpanda.remote.write=%t" $values.connectors.storage.remote.write.status) (printf "offset.storage.replication.factor=%d" ($values.connectors.storage.replicationFactor.offset | int)) (printf "config.storage.replication.factor=%d" ($values.connectors.storage.replicationFactor.config | int)) (printf "status.storage.replication.factor=%d" ($values.connectors.storage.replicationFactor.status | int)) (printf "producer.linger.ms=%d" ($values.connectors.producerLingerMS | int)) (printf "producer.batch.size=%d" ($values.connectors.producerBatchSize | int)) "config.providers=file,secretsManager,env" "config.providers.file.class=org.apache.kafka.common.config.provider.FileConfigProvider") -}} +{{- if $values.connectors.secretManager.enabled -}} +{{- $lines = (concat (default (list ) $lines) (list "config.providers.secretsManager.class=com.github.jcustenborder.kafka.config.aws.SecretsManagerConfigProvider" (printf "config.providers.secretsManager.param.secret.prefix=%s%s" $values.connectors.secretManager.consolePrefix $values.connectors.secretManager.connectorsPrefix) (printf "config.providers.secretsManager.param.aws.region=%s" $values.connectors.secretManager.region))) -}} +{{- end -}} +{{- $lines = (concat (default (list ) $lines) (list "config.providers.env.class=org.apache.kafka.common.config.provider.EnvVarConfigProvider")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (join "\n" $lines)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.volumes" -}} +{{- $values := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $volumes := (coalesce nil) -}} +{{- if (not (empty $values.connectors.brokerTLS.ca.secretRef)) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "defaultMode" (0o444 | int) "secretName" $values.connectors.brokerTLS.ca.secretRef )) )) (dict "name" "truststore" )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.cert.secretRef)) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "defaultMode" (0o444 | int) "secretName" $values.connectors.brokerTLS.cert.secretRef )) )) (dict "name" "cert" )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.key.secretRef)) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "defaultMode" (0o444 | int) "secretName" $values.connectors.brokerTLS.key.secretRef )) )) (dict "name" "key" )))) -}} +{{- end -}} +{{- if (get (fromJson (include "connectors.Auth.SASLEnabled" (dict "a" (list $values.auth) ))) "r") -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "defaultMode" (0o444 | int) "secretName" $values.auth.sasl.secretRef )) )) (dict "name" "rc-credentials" )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $volumes) (default (list ) $values.storage.volume))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.volumeMountss" -}} +{{- $values := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $mounts := (coalesce nil) -}} +{{- if (get (fromJson (include "connectors.Auth.SASLEnabled" (dict "a" (list $values.auth) ))) "r") -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "mountPath" "/opt/kafka/connect-password/rc-credentials" "name" "rc-credentials" )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.ca.secretRef)) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "truststore" "mountPath" "/opt/kafka/connect-certs/ca" )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.cert.secretRef)) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "cert" "mountPath" "/opt/kafka/connect-certs/cert" )))) -}} +{{- end -}} +{{- if (not (empty $values.connectors.brokerTLS.key.secretRef)) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "key" "mountPath" "/opt/kafka/connect-certs/key" )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $mounts) (default (list ) $values.storage.volumeMounts))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.go.tpl new file mode 100644 index 0000000000..49b7115382 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.go.tpl @@ -0,0 +1,131 @@ +{{- /* Generated from "helpers.go" */ -}} + +{{- define "connectors.Name" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $name := (default $dot.Chart.Name $values.nameOverride) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "connectors.trunc" (dict "a" (list $name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.Fullname" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (empty $values.fullnameOverride)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "connectors.trunc" (dict "a" (list $values.fullnameOverride) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $name := (default $dot.Chart.Name $values.nameOverride) -}} +{{- if (contains $name $dot.Release.Name) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "connectors.trunc" (dict "a" (list $dot.Release.Name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "connectors.trunc" (dict "a" (list (printf "%s-%s" $dot.Release.Name $name)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.FullLabels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) (dict "helm.sh/chart" (get (fromJson (include "connectors.Chart" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/managed-by" $dot.Release.Service ) (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.PodLabels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) (dict "app.kubernetes.io/name" (get (fromJson (include "connectors.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name "app.kubernetes.io/component" (get (fromJson (include "connectors.Name" (dict "a" (list $dot) ))) "r") ) $values.commonLabels)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.Chart" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $chart := (printf "%s-%s" $dot.Chart.Name $dot.Chart.Version) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "connectors.trunc" (dict "a" (list (replace "+" "_" $chart)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.Semver" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (trimPrefix "v" (get (fromJson (include "connectors.Tag" (dict "a" (list $dot) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.ServiceAccountName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if $values.serviceAccount.create -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default (get (fromJson (include "connectors.Fullname" (dict "a" (list $dot) ))) "r") $values.serviceAccount.name)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default "default" $values.serviceAccount.name)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.ServiceName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default (get (fromJson (include "connectors.Fullname" (dict "a" (list $dot) ))) "r") $values.service.name)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.Tag" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tag := (default $dot.Chart.AppVersion $values.image.tag) -}} +{{- $matchString := "^v(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" -}} +{{- if (not (mustRegexMatch $matchString $tag)) -}} +{{- $_ := (fail "image.tag must start with a 'v' and be a valid semver") -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tag) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "connectors.trunc" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (trimSuffix "-" (trunc (63 | int) $s))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.tpl new file mode 100644 index 0000000000..89c888eeef --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_helpers.tpl @@ -0,0 +1,79 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{/* +Expand the name of the chart. +*/}} +{{- define "connectors.name" -}} +{{- get ((include "connectors.Name" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "connectors.fullname" }} +{{- get ((include "connectors.Fullname" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +full helm labels + common labels +*/}} +{{- define "full.labels" -}} +{{- (get ((include "connectors.FullLabels" (dict "a" (list .))) | fromJson) "r") | toYaml }} +{{- end -}} + +{{/* +pod labels merged with common labels +*/}} +{{- define "connectors-pod-labels" -}} +{{- (get ((include "connectors.PodLabels" (dict "a" (list .))) | fromJson) "r") | toYaml }} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "connectors.chart" -}} +{{- get ((include "connectors.Chart" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Get the version of redpanda being used as an image +*/}} +{{- define "connectors.semver" -}} +{{- get ((include "connectors.Tag" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "connectors.serviceAccountName" -}} +{{- get ((include "connectors.ServiceAccountName" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Create the name of the service to use +*/}} +{{- define "connectors.serviceName" -}} +{{- get ((include "connectors.ServiceName" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Use AppVersion if image.tag is not set +*/}} +{{- define "connectors.tag" -}} +{{- get ((include "connectors.Tag" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_pod-monitor.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_pod-monitor.go.tpl new file mode 100644 index 0000000000..4e12b20084 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_pod-monitor.go.tpl @@ -0,0 +1,18 @@ +{{- /* Generated from "podmonitor.go" */ -}} + +{{- define "connectors.PodMonitor" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.monitoring.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "podMetricsEndpoints" (coalesce nil) "selector" (dict ) "namespaceSelector" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "monitoring.coreos.com/v1" "kind" "PodMonitor" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "connectors.Fullname" (dict "a" (list $dot) ))) "r") "labels" $values.monitoring.labels "annotations" $values.monitoring.annotations )) "spec" (mustMergeOverwrite (dict "podMetricsEndpoints" (coalesce nil) "selector" (dict ) "namespaceSelector" (dict ) ) (dict "namespaceSelector" $values.monitoring.namespaceSelector "podMetricsEndpoints" (list (mustMergeOverwrite (dict "bearerTokenSecret" (dict "key" "" ) ) (dict "path" "/" "port" "prometheus" ))) "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") )) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_service.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_service.go.tpl new file mode 100644 index 0000000000..54a7ce8a05 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_service.go.tpl @@ -0,0 +1,20 @@ +{{- /* Generated from "service.go" */ -}} + +{{- define "connectors.Service" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $ports := (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "rest-api" "port" ($values.connectors.restPort | int) "targetPort" ($values.connectors.restPort | int) "protocol" "TCP" ))) -}} +{{- range $_, $port := $values.service.ports -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" $port.name "port" ($port.port | int) "targetPort" ($port.port | int) "protocol" "TCP" )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Service" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "connectors.ServiceName" (dict "a" (list $dot) ))) "r") "labels" (merge (dict ) (get (fromJson (include "connectors.FullLabels" (dict "a" (list $dot) ))) "r") $values.service.annotations) )) "spec" (mustMergeOverwrite (dict ) (dict "ipFamilies" (list "IPv4") "ipFamilyPolicy" "SingleStack" "ports" $ports "selector" (get (fromJson (include "connectors.PodLabels" (dict "a" (list $dot) ))) "r") "sessionAffinity" "None" "type" "ClusterIP" )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_serviceaccount.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_serviceaccount.go.tpl new file mode 100644 index 0000000000..31b5ac2acd --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_serviceaccount.go.tpl @@ -0,0 +1,18 @@ +{{- /* Generated from "serviceaccount.go" */ -}} + +{{- define "connectors.ServiceAccount" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.serviceAccount.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "ServiceAccount" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "annotations" $values.serviceAccount.annotations "labels" (get (fromJson (include "connectors.FullLabels" (dict "a" (list $dot) ))) "r") "name" (get (fromJson (include "connectors.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_shims.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_shims.tpl new file mode 100644 index 0000000000..e3bb40e415 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_shims.tpl @@ -0,0 +1,289 @@ +{{- /* Generated from "bootstrap.go" */ -}} + +{{- define "_shims.typetest" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs $typ $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.typeassertion" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not (typeIs $typ $value)) -}} +{{- $_ := (fail (printf "expected type of %q got: %T" $typ $value)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $value) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.dicttest" -}} +{{- $m := (index .a 0) -}} +{{- $key := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (hasKey $m $key) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (index $m $key) true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.compact" -}} +{{- $args := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $out := (dict ) -}} +{{- range $i, $e := $args -}} +{{- $_ := (set $out (printf "T%d" ((add (1 | int) $i) | int)) $e) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $out) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.deref" -}} +{{- $ptr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq $ptr (coalesce nil)) -}} +{{- $_ := (fail "nil dereference") -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.len" -}} +{{- $m := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq $m (coalesce nil)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (0 | int)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (len $m)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Deref" -}} +{{- $ptr := (index .a 0) -}} +{{- $def := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne $ptr (coalesce nil)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $def) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Equal" -}} +{{- $a := (index .a 0) -}} +{{- $b := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (and (eq $a (coalesce nil)) (eq $b (coalesce nil))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (eq $a $b)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.lookup" -}} +{{- $apiVersion := (index .a 0) -}} +{{- $kind := (index .a 1) -}} +{{- $namespace := (index .a 2) -}} +{{- $name := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (lookup $apiVersion $kind $namespace $name) -}} +{{- if (empty $result) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (coalesce nil) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $result true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asnumeric" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asintegral" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (or (typeIs "int64" $value) (typeIs "int" $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (and (typeIs "float64" $value) (eq (floor $value) $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.parseResource" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $repr) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (float64 $repr) 1.0)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (not (typeIs "string" $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity expected string or float64 got: %T (%v)" $repr $repr)) -}} +{{- end -}} +{{- if (not (regexMatch `^[0-9]+(\.[0-9]{0,6})?(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$` $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity: %q" $repr)) -}} +{{- end -}} +{{- $reprStr := (toString $repr) -}} +{{- $unit := (regexFind "(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)$" $repr) -}} +{{- $numeric := (float64 (substr (0 | int) ((sub ((get (fromJson (include "_shims.len" (dict "a" (list $reprStr) ))) "r") | int) ((get (fromJson (include "_shims.len" (dict "a" (list $unit) ))) "r") | int)) | int) $reprStr)) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list (dict "" 1.0 "m" 0.001 "k" (1000 | int) "M" (1000000 | int) "G" (1000000000 | int) "T" (1000000000000 | int) "P" (1000000000000000 | int) "Ki" (1024 | int) "Mi" (1048576 | int) "Gi" (1073741824 | int) "Ti" (1099511627776 | int) "Pi" (1125899906842624 | int) ) $unit (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_1.T2 -}} +{{- $scale := ($tmp_tuple_1.T1 | float64) -}} +{{- if (not $ok) -}} +{{- $_ := (fail (printf "unknown unit: %q" $unit)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $numeric $scale)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MustParse" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_2.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_2.T1 | float64) -}} +{{- $strs := (list "" "m" "k" "M" "G" "T" "P" "Ki" "Mi" "Gi" "Ti" "Pi") -}} +{{- $scales := (list 1.0 0.001 (1000 | int) (1000000 | int) (1000000000 | int) (1000000000000 | int) (1000000000000000 | int) (1024 | int) (1048576 | int) (1073741824 | int) (1099511627776 | int) (1125899906842624 | int)) -}} +{{- $idx := -1 -}} +{{- range $i, $s := $scales -}} +{{- if (eq ($s | float64) ($scale | float64)) -}} +{{- $idx = $i -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (eq $idx -1) -}} +{{- $_ := (fail (printf "unknown scale: %v" $scale)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s%s" (toString $numeric) (index $strs $idx))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_Value" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_3.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_3.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf $numeric $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MilliValue" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_4.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_4.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf ((mulf $numeric 1000.0) | float64) $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.render-manifest" -}} +{{- $tpl := (index . 0) -}} +{{- $dot := (index . 1) -}} +{{- $manifests := (get ((include $tpl (dict "a" (list $dot))) | fromJson) "r") -}} +{{- if not (typeIs "[]interface {}" $manifests) -}} +{{- $manifests = (list $manifests) -}} +{{- end -}} +{{- range $_, $manifest := $manifests -}} +{{- if ne $manifest nil }} +--- +{{toYaml (unset (unset $manifest "status") "creationTimestamp")}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_values.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_values.go.tpl new file mode 100644 index 0000000000..9b304d4bf6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/_values.go.tpl @@ -0,0 +1,15 @@ +{{- /* Generated from "values.go" */ -}} + +{{- define "connectors.Auth.SASLEnabled" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $saslEnabled := (not (empty $c.sasl.userName)) -}} +{{- $saslEnabled = (and $saslEnabled (not (empty $c.sasl.mechanism))) -}} +{{- $saslEnabled = (and $saslEnabled (not (empty $c.sasl.secretRef))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $saslEnabled) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/deployment.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/deployment.yaml new file mode 100644 index 0000000000..ee78b69ebf --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/deployment.yaml @@ -0,0 +1,17 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "connectors.Deployment" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/pod-monitor.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/pod-monitor.yaml new file mode 100644 index 0000000000..42c1457546 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/pod-monitor.yaml @@ -0,0 +1,17 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "connectors.PodMonitor" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/service.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/service.yaml new file mode 100644 index 0000000000..0b8825befc --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/service.yaml @@ -0,0 +1,17 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "connectors.Service" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/serviceaccount.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/serviceaccount.yaml new file mode 100644 index 0000000000..eda755fb14 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/serviceaccount.yaml @@ -0,0 +1,17 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "connectors.ServiceAccount" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/tests/01-mm2-values.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/tests/01-mm2-values.yaml new file mode 100644 index 0000000000..c369806c8b --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/templates/tests/01-mm2-values.yaml @@ -0,0 +1,176 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- define "curl-options" -}} +{{- print " -svm3 --fail --retry \"120\" --retry-max-time \"120\" --retry-all-errors -o - -w \"\\nstatus=%{http_code} %{redirect_url} size=%{size_download} time=%{time_total} content-type=\\\"%{content_type}\\\"\\n\" "}} +{{- end -}} +{{- if .Values.test.create -}} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "connectors.fullname" . }}-mm2-test + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: create-mm2 + image: docker.redpanda.com/redpandadata/redpanda:latest + command: + - /bin/bash + - -c + - | + set -xe + + trap connectorsState ERR + + connectorsState () { + echo check connectors expand status + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors?expand=status + echo check connectors expand info + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors?expand=info + echo check connector configuration + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors/$CONNECTOR_NAME + echo check connector topics + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors/$CONNECTOR_NAME/topics + } + + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors + + SASL_MECHANISM="PLAIN" + {{- if .Values.auth.sasl.enabled }} + set -e + set +x + + IFS=: read -r CONNECT_SASL_USERNAME KAFKA_SASL_PASSWORD CONNECT_SASL_MECHANISM < $(find /mnt/users/* -print) + CONNECT_SASL_MECHANISM=${CONNECT_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + if [[ -n "$CONNECT_SASL_USERNAME" && -n "$KAFKA_SASL_PASSWORD" && -n "$CONNECT_SASL_MECHANISM" ]]; then + rpk profile set user=$CONNECT_SASL_USERNAME pass=$KAFKA_SASL_PASSWORD sasl.mechanism=$CONNECT_SASL_MECHANISM + SASL_MECHANISM=$CONNECT_SASL_MECHANISM + JAAS_CONFIG_SOURCE="\"source.cluster.sasl.jaas.config\": \"org.apache.kafka.common.security.scram.ScramLoginModule required username=\\\\"\"${CONNECT_SASL_USERNAME}\\\\"\" password=\\\\"\"${KAFKA_SASL_PASSWORD}\\\\"\";\"," + JAAS_CONFIG_TARGET="\"target.cluster.sasl.jaas.config\": \"org.apache.kafka.common.security.scram.ScramLoginModule required username=\\\\"\"${CONNECT_SASL_USERNAME}\\\\"\" password=\\\\"\"${KAFKA_SASL_PASSWORD}\\\\"\";\"," + fi + + set -x + set +e + {{- end }} + + rpk profile create test + rpk profile set tls.enabled={{.Values.connectors.brokerTLS.enabled}} brokers={{ .Values.connectors.bootstrapServers }} + {{- if .Values.connectors.brokerTLS.ca.secretRef }} + rpk profile set tls.ca={{ printf "/redpanda-certs/%s" (default "ca.crt" .Values.connectors.brokerTLS.ca.secretNameOverwrite) }} + {{- end }} + + {{- if .Values.connectors.brokerTLS.enabled }} + CONNECT_TLS_ENABLED=true + {{- else }} + CONNECT_TLS_ENABLED=false + {{- end }} + SECURITY_PROTOCOL=PLAINTEXT + if [[ -n "$CONNECT_SASL_MECHANISM" && $CONNECT_TLS_ENABLED == "true" ]]; then + SECURITY_PROTOCOL="SASL_SSL" + elif [[ -n "$CONNECT_SASL_MECHANISM" ]]; then + SECURITY_PROTOCOL="SASL_PLAINTEXT" + elif [[ $CONNECT_TLS_ENABLED == "true" ]]; then + SECURITY_PROTOCOL="SSL" + fi + + rpk topic list + rpk topic create test-topic + rpk topic list + echo "Test message!" | rpk topic produce test-topic + + CONNECTOR_NAME=mm2-$RANDOM + cat << 'EOF' > /tmp/mm2-conf.json + { + "name": "CONNECTOR_NAME", + "config": { + "connector.class": "org.apache.kafka.connect.mirror.MirrorSourceConnector", + "topics": "test-topic", + "replication.factor": "1", + "tasks.max": "1", + "source.cluster.bootstrap.servers": {{ .Values.connectors.bootstrapServers | quote }}, + "target.cluster.bootstrap.servers": {{ .Values.connectors.bootstrapServers | quote }}, + "target.cluster.alias": "test-only", + "source.cluster.alias": "source", + "key.converter": "org.apache.kafka.connect.converters.ByteArrayConverter", + "value.converter": "org.apache.kafka.connect.converters.ByteArrayConverter", + "source->target.enabled": "true", + "target->source.enabled": "false", + "sync.topic.configs.interval.seconds": "5", + "sync.topics.configs.enabled": "true", + "source.cluster.ssl.truststore.type": "PEM", + "target.cluster.ssl.truststore.type": "PEM", + "source.cluster.ssl.truststore.location": {{ printf "/opt/kafka/connect-certs/ca/%s" (default "ca.crt" .Values.connectors.brokerTLS.ca.secretNameOverwrite) | quote }}, + "target.cluster.ssl.truststore.location": {{ printf "/opt/kafka/connect-certs/ca/%s" (default "ca.crt" .Values.connectors.brokerTLS.ca.secretNameOverwrite) | quote }}, + JAAS_CONFIG_SOURCE + JAAS_CONFIG_TARGET + "source.cluster.security.protocol": "SECURITY_PROTOCOL", + "target.cluster.security.protocol": "SECURITY_PROTOCOL", + "source.cluster.sasl.mechanism": "SASL_MECHANISM", + "target.cluster.sasl.mechanism": "SASL_MECHANISM", + "offset-syncs.topic.replication.factor": 1 + } + } + EOF + + sed -i "s/CONNECTOR_NAME/$CONNECTOR_NAME/g" /tmp/mm2-conf.json + sed -i "s/SASL_MECHANISM/$SASL_MECHANISM/g" /tmp/mm2-conf.json + sed -i "s/SECURITY_PROTOCOL/$SECURITY_PROTOCOL/g" /tmp/mm2-conf.json + set +x + sed -i "s/JAAS_CONFIG_SOURCE/$JAAS_CONFIG_SOURCE/g" /tmp/mm2-conf.json + sed -i "s/JAAS_CONFIG_TARGET/$JAAS_CONFIG_TARGET/g" /tmp/mm2-conf.json + set -x + + curl {{ template "curl-options" . }} -H 'Content-Type: application/json' http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors -d @/tmp/mm2-conf.json + + # The rpk topic consume could fail for the first few times as kafka connect needs + # to spawn the task and copy one message from the source topic. To solve this race condition + # the retry should be implemented in bash for rpk topic consume or other mechanism that + # can confirm source connectors started its execution. As a fast fix fixed 30 second fix is added. + sleep 30 + + rpk topic consume source.test-topic -n 1 | grep "Test message!" + + curl {{ template "curl-options" . }} -X DELETE http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors/$CONNECTOR_NAME + + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" . }}:{{ .Values.connectors.restPort }}/connectors + + rpk topic delete test-topic source.test-topic mm2-offset-syncs.test-only.internal + volumeMounts: + {{- if .Values.connectors.brokerTLS.ca.secretRef }} + - mountPath: /redpanda-certs + name: redpanda-ca + {{- end }} + {{- toYaml .Values.storage.volumeMounts | nindent 8 }} + volumes: + {{- if .Values.connectors.brokerTLS.ca.secretRef }} + - name: redpanda-ca + secret: + defaultMode: 0444 + secretName: {{ .Values.connectors.brokerTLS.ca.secretRef }} + {{- end }} + {{- toYaml .Values.storage.volume | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/charts/connectors/values.yaml b/charts/redpanda/redpanda/5.9.10/charts/connectors/values.yaml new file mode 100644 index 0000000000..f230a84d37 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/connectors/values.yaml @@ -0,0 +1,311 @@ +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# This file contains values for variables referenced from yaml files in the templates directory. +# +# For further information on Helm templating see the documentation at: +# https://helm.sh/docs/chart_template_guide/values_files/ + +# +# >>> This chart requires Helm version 3.6.0 or greater <<< +# + +# Common settings +# +# -- Override `connectors.name` template. +nameOverride: "" +# -- Override `connectors.fullname` template. +fullnameOverride: "" +# -- Additional labels to add to all Kubernetes objects. +# For example, `my.k8s.service: redpanda`. +commonLabels: {} +# -- Taints to be tolerated by Pods. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). +tolerations: [] + +# -- Redpanda Docker image settings. +image: + # -- Docker repository from which to pull the Redpanda Docker image. + repository: docker.redpanda.com/redpandadata/connectors + # -- The Redpanda version. + # See DockerHub for: + # [All stable versions](https://hub.docker.com/r/redpandadata/redpanda/tags) + # and [all unstable versions](https://hub.docker.com/r/redpandadata/redpanda-unstable/tags). + # @default -- `Chart.appVersion`. + tag: "" + # -- The imagePullPolicy. + # If `image.tag` is 'latest', the default is `Always`. + pullPolicy: IfNotPresent + +# -- Pull secrets may be used to provide credentials to image repositories +# See https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ +imagePullSecrets: [] + +test: + create: true + +connectors: + # -- The port on which the Kafka Connect REST API listens. The API is used for administrative tasks. + restPort: 8083 + # -- A comma-separated list of Redpanda broker addresses in the format of IP:Port or DNS:Port. Kafka Connect uses this to connect to the Redpanda/Kafka cluster. + bootstrapServers: "" + # A comma-separated list of Schema Registry addresses in the format IP:Port or DNS:Port. The Schema Registry is a service that manages the schemas used by producers and consumers. + schemaRegistryURL: "" + # -- A placeholder for any Java configuration settings for Kafka Connect that are not explicitly defined in this Helm chart. Java configuration settings are passed to the Kafka Connect startup script. + additionalConfiguration: "" + secretManager: + enabled: false + region: "" + consolePrefix: "" + connectorsPrefix: "" + # -- The number of bytes of records a producer will attempt to batch together before sending to Redpanda. Batching improves throughput. + producerBatchSize: 131072 + # -- The time, in milliseconds, that a producer will wait before sending a batch of records. Waiting allows the producer to gather more records in the same batch and improve throughput. + producerLingerMS: 1 + storage: + # -- The number of replicas for each of the internal topics that Kafka Connect uses. + replicationFactor: + # -- Replication factor for the offset topic. + offset: -1 + # -- Replication factor for the configuration topic. + config: -1 + # -- Replication factor for the status topic. + status: -1 + # -- Indicates if read and write operations for the respective topics are allowed remotely. + remote: + read: + offset: false + config: false + status: false + write: + offset: false + config: false + status: false + topic: + # -- The name of the internal topic that Kafka Connect uses to store source connector offsets. + offset: _internal_connectors_offsets + # -- The name of the internal topic that Kafka Connect uses to store connector and task configurations. + config: _internal_connectors_configs + # -- The name of the internal topic that Kafka Connect uses to store connector and task status updates. + status: _internal_connectors_status + # -- A unique string that identifies the Kafka Connect cluster. It's used in the formation of the internal topic names, ensuring that multiple Kafka Connect clusters can connect to the same Redpanda cluster without interfering with each other. + groupID: connectors-cluster + brokerTLS: + enabled: false + ca: + # -- The name of the Secret where the ca.crt file content is located. + secretRef: "" + # -- If `secretRef` points to a Secret where the certificate authority (CA) is not under the + # `ca.crt` key, use `secretNameOverwrite` to overwrite it e.g. `corp-ca.crt`. + secretNameOverwrite: "" + cert: + # -- The name of the secret where client signed certificate is located + secretRef: "" + # -- If secretRef points to secret where client signed certificate is not under + # tls.crt key then please use secretNameOverwrite to overwrite it e.g. corp-tls.crt + secretNameOverwrite: "" + key: + # -- The name of the secret where client private key is located + secretRef: "" + # -- If secretRef points to secret where client private key is not under + # tls.key key then please use secretNameOverwrite to overwrite it e.g. corp-tls.key + secretNameOverwrite: "" + +# -- Authentication settings. +# For details, +# see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/). +# The first line of the secret file is used. So the first superuser is used to authenticate to the Redpanda cluster. +auth: + sasl: + enabled: false + # -- The authentication mechanism to use for the superuser. Options are `scram-sha-256` and `scram-sha-512`. + mechanism: scram-sha-512 + # -- A Secret that contains your SASL user password. + secretRef: "" + userName: "" + +# -- Log-level settings. +logging: + # -- Log level + # Valid values (from least to most verbose) are: `error`, `warn`, `info` and `debug`. + level: warn + +# -- Monitoring. +# When set to `true`, the Helm chart creates a PodMonitor that can be used by Prometheus-Operator or VictoriaMetrics-Operator to scrape the metrics. +monitoring: + enabled: false + scrapeInterval: 30s + labels: {} + annotations: {} + namespaceSelector: + any: true + +container: + # + # -- Security context for the Redpanda Connectors container. + # See also `deployment.securityContext` for Pod-level settings. + securityContext: + allowPrivilegeEscalation: false + # -- Pod resource management. + resources: + request: + # Numeric values here are also acceptable. + cpu: "1" + memory: 2350Mi + limits: + cpu: "1" + memory: 2350Mi + # -- Java maximum heap size must not be greater than `container.resources.limits.memory`. + javaMaxHeapSize: 2G + javaGCLogEnabled: "false" + +deployment: + # Replicas can be used to scale Deployment + # replicas + + create: true + # Customize the command to use as the entrypoint of the Deployment. + # command: [] + strategy: + type: RollingUpdate + schedulerName: "" + budget: + maxUnavailable: 1 + # -- Additional annotations to apply to the Pods of this Deployment. + annotations: {} + # -- Adjust the period for your probes to meet your needs. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). + livenessProbe: + initialDelaySeconds: 10 + failureThreshold: 3 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + readinessProbe: + initialDelaySeconds: 60 + failureThreshold: 2 + periodSeconds: 10 + successThreshold: 3 + timeoutSeconds: 5 + + # -- Additional environment variables for the Pods. + extraEnv: [] + # - name: RACK_ID + # value: "1" + + # -- Configure extra environment variables from Secrets and ConfigMaps. + extraEnvFrom: [] + # - secretRef: + # name: my-secret + # - configMapRef: + # name: my-configmap + + # -- The maximum time in seconds for a deployment to make progress before it is + # considered to be failed. The deployment controller will continue to process + # failed deployments and a condition with a ProgressDeadlineExceeded reason + # will be surfaced in the deployment status. Note that progress will not be + # estimated during the time a deployment is paused. + progressDeadlineSeconds: 600 + + # -- The number of old ReplicaSets to retain to allow rollback. This is a pointer + # to distinguish between explicit zero and not specified. + revisionHistoryLimit: 10 + + # -- Inter-Pod Affinity rules for scheduling Pods of this Deployment. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + podAffinity: {} + # -- Node Affinity rules for scheduling Pods of this Deployment. + # The suggestion would be to spread Pods according to topology zone. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity). + nodeAffinity: {} + # -- Anti-affinity rules for scheduling Pods of this Deployment. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + # You may either edit the default settings for anti-affinity rules, + # or specify new anti-affinity rules to use instead of the defaults. + podAntiAffinity: + # -- The `topologyKey` to be used. + # Can be used to spread across different nodes, AZs, regions etc. + topologyKey: kubernetes.io/hostname + # -- Valid anti-affinity types are `soft`, `hard`, or `custom`. + # Use `custom` if you want to supply your own anti-affinity rules in the `podAntiAffinity.custom` object. + type: hard + # -- Weight for `soft` anti-affinity rules. + # Does not apply for other anti-affinity types. + weight: 100 + # -- Change `podAntiAffinity.type` to `custom` and provide your own podAntiAffinity rules here. + custom: {} + # -- Node selection constraints for scheduling Pods of this Deployment. + # These constraints override the global `nodeSelector` value. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). + nodeSelector: {} + # -- PriorityClassName given to Pods of this Deployment. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). + priorityClassName: "" + # -- Taints to be tolerated by Pods of this Deployment. + # These tolerations override the global tolerations value. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + tolerations: [] + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/). + topologySpreadConstraints: + - maxSkew: 1 + topologyKey: topology.kubernetes.io/zone + whenUnsatisfiable: ScheduleAnyway + securityContext: + fsGroup: 101 + runAsUser: 101 + fsGroupChangePolicy: OnRootMismatch + terminationGracePeriodSeconds: 30 + restartPolicy: Always + +storage: + volume: + - emptyDir: + medium: Memory + sizeLimit: 5Mi + name: rp-connect-tmp + volumeMounts: + - mountPath: /tmp + name: rp-connect-tmp + +# -- ServiceAccount management. +serviceAccount: + # -- Specifies whether a ServiceAccount should be created. + create: false + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- The name of the ServiceAccount to use. + # If not set and `serviceAccount.create` is `true`, + # a name is generated using the `connectors.fullname` template. + name: "" + +# -- Service management. +service: + # -- Annotations to add to the Service. + annotations: {} + # -- The name of the service to use. + # If not set, a name is generated using the `connectors.fullname` template. + name: "" + ports: + - name: prometheus + port: 9404 diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/.helmignore b/charts/redpanda/redpanda/5.9.10/charts/console/.helmignore new file mode 100644 index 0000000000..d5bb5e6ba6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/.helmignore @@ -0,0 +1,28 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +README.md.gotmpl +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +*.go +testdata/ +ci/ diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/Chart.yaml b/charts/redpanda/redpanda/5.9.10/charts/console/Chart.yaml new file mode 100644 index 0000000000..37a546db95 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/Chart.yaml @@ -0,0 +1,23 @@ +annotations: + artifacthub.io/images: | + - name: redpanda + image: docker.redpanda.com/redpandadata/console:v2.7.2 + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Documentation + url: https://docs.redpanda.com + - name: "Helm (>= 3.6.0)" + url: https://helm.sh/docs/intro/install/ +apiVersion: v2 +appVersion: v2.7.2 +description: Helm chart to deploy Redpanda Console. +icon: https://images.ctfassets.net/paqvtpyf8rwu/3cYHw5UzhXCbKuR24GDFGO/73fb682e6157d11c10d5b2b5da1d5af0/skate-stand-panda.svg +kubeVersion: '>= 1.25.0-0' +maintainers: +- name: redpanda-data + url: https://github.com/orgs/redpanda-data/people +name: console +sources: +- https://github.com/redpanda-data/helm-charts +type: application +version: 0.7.30 diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/README.md b/charts/redpanda/redpanda/5.9.10/charts/console/README.md new file mode 100644 index 0000000000..9fb3932733 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/README.md @@ -0,0 +1,353 @@ +# Redpanda Console Helm Chart Specification +--- +description: Find the default values and descriptions of settings in the Redpanda Console Helm chart. +--- + +![Version: 0.7.30](https://img.shields.io/badge/Version-0.7.30-informational?style=flat-square) ![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![AppVersion: v2.7.2](https://img.shields.io/badge/AppVersion-v2.7.2-informational?style=flat-square) + +This page describes the official Redpanda Console Helm Chart. In particular, this page describes the contents of the chart’s [`values.yaml` file](https://github.com/redpanda-data/helm-charts/blob/main/charts/console/values.yaml). +Each of the settings is listed and described on this page, along with any default values. + +The Redpanda Console Helm chart is included as a subchart in the Redpanda Helm chart so that you can deploy and configure Redpanda and Redpanda Console together. +For instructions on how to install and use the chart, refer to the [deployment documentation](https://docs.redpanda.com/docs/deploy/deployment-option/self-hosted/kubernetes/kubernetes-deploy/). +For instructions on how to override and customize the chart’s values, see [Configure Redpanda Console](https://docs.redpanda.com/docs/manage/kubernetes/configure-helm-chart/#configure-redpanda-console). + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.13.1](https://github.com/norwoodj/helm-docs/releases/v1.13.1) + +## Source Code + +* + +## Requirements + +Kubernetes: `>= 1.25.0-0` + +## Settings + +### [affinity](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=affinity) + +**Default:** `{}` + +### [annotations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=annotations) + +Annotations to add to the deployment. + +**Default:** `{}` + +### [automountServiceAccountToken](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=automountServiceAccountToken) + +Automount API credentials for the Service Account into the pod. + +**Default:** `true` + +### [autoscaling.enabled](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=autoscaling.enabled) + +**Default:** `false` + +### [autoscaling.maxReplicas](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=autoscaling.maxReplicas) + +**Default:** `100` + +### [autoscaling.minReplicas](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=autoscaling.minReplicas) + +**Default:** `1` + +### [autoscaling.targetCPUUtilizationPercentage](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=autoscaling.targetCPUUtilizationPercentage) + +**Default:** `80` + +### [commonLabels](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=commonLabels) + +**Default:** `{}` + +### [configmap.create](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=configmap.create) + +**Default:** `true` + +### [console.config](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=console.config) + +Settings for the `Config.yaml` (required). For a reference of configuration settings, see the [Redpanda Console documentation](https://docs.redpanda.com/docs/reference/console/config/). + +**Default:** `{}` + +### [deployment.create](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=deployment.create) + +**Default:** `true` + +### [enterprise](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=enterprise) + +Settings for license key, as an alternative to secret.enterprise when a license secret is available + +**Default:** + +``` +{"licenseSecretRef":{"key":"","name":""}} +``` + +### [extraContainers](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=extraContainers) + +Add additional containers, such as for oauth2-proxy. + +**Default:** `[]` + +### [extraEnv](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=extraEnv) + +Additional environment variables for the Redpanda Console Deployment. + +**Default:** `[]` + +### [extraEnvFrom](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=extraEnvFrom) + +Additional environment variables for Redpanda Console mapped from Secret or ConfigMap. + +**Default:** `[]` + +### [extraVolumeMounts](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=extraVolumeMounts) + +Add additional volume mounts, such as for TLS keys. + +**Default:** `[]` + +### [extraVolumes](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=extraVolumes) + +Add additional volumes, such as for TLS keys. + +**Default:** `[]` + +### [fullnameOverride](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=fullnameOverride) + +Override `console.fullname` template. + +**Default:** `""` + +### [image](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=image) + +Redpanda Console Docker image settings. + +**Default:** + +``` +{"pullPolicy":"IfNotPresent","registry":"docker.redpanda.com","repository":"redpandadata/console","tag":""} +``` + +### [image.pullPolicy](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=image.pullPolicy) + +The imagePullPolicy. + +**Default:** `"IfNotPresent"` + +### [image.repository](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=image.repository) + +Docker repository from which to pull the Redpanda Docker image. + +**Default:** `"redpandadata/console"` + +### [image.tag](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=image.tag) + +The Redpanda Console version. See DockerHub for: [All stable versions](https://hub.docker.com/r/redpandadata/console/tags) and [all unstable versions](https://hub.docker.com/r/redpandadata/console-unstable/tags). + +**Default:** `Chart.appVersion` + +### [imagePullSecrets](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=imagePullSecrets) + +Pull secrets may be used to provide credentials to image repositories See https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + +**Default:** `[]` + +### [ingress.annotations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.annotations) + +**Default:** `{}` + +### [ingress.className](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.className) + +**Default:** `nil` + +### [ingress.enabled](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.enabled) + +**Default:** `false` + +### [ingress.hosts[0].host](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.hosts[0].host) + +**Default:** `"chart-example.local"` + +### [ingress.hosts[0].paths[0].path](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.hosts[0].paths[0].path) + +**Default:** `"/"` + +### [ingress.hosts[0].paths[0].pathType](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.hosts[0].paths[0].pathType) + +**Default:** `"ImplementationSpecific"` + +### [ingress.tls](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=ingress.tls) + +**Default:** `[]` + +### [initContainers](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=initContainers) + +Any initContainers defined should be written here + +**Default:** `{"extraInitContainers":""}` + +### [initContainers.extraInitContainers](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=initContainers.extraInitContainers) + +Additional set of init containers + +**Default:** `""` + +### [livenessProbe](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=livenessProbe) + +Settings for liveness and readiness probes. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes). + +**Default:** + +``` +{"failureThreshold":3,"periodSeconds":10,"successThreshold":1,"timeoutSeconds":1} +``` + +### [nameOverride](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=nameOverride) + +Override `console.name` template. + +**Default:** `""` + +### [nodeSelector](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=nodeSelector) + +**Default:** `{}` + +### [podAnnotations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=podAnnotations) + +**Default:** `{}` + +### [podLabels](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=podLabels) + +**Default:** `{}` + +### [podSecurityContext.fsGroup](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=podSecurityContext.fsGroup) + +**Default:** `99` + +### [podSecurityContext.runAsUser](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=podSecurityContext.runAsUser) + +**Default:** `99` + +### [priorityClassName](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=priorityClassName) + +PriorityClassName given to Pods. For details, see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). + +**Default:** `""` + +### [readinessProbe.failureThreshold](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=readinessProbe.failureThreshold) + +**Default:** `3` + +### [readinessProbe.initialDelaySeconds](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=readinessProbe.initialDelaySeconds) + +Grant time to test connectivity to upstream services such as Kafka and Schema Registry. + +**Default:** `10` + +### [readinessProbe.periodSeconds](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=readinessProbe.periodSeconds) + +**Default:** `10` + +### [readinessProbe.successThreshold](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=readinessProbe.successThreshold) + +**Default:** `1` + +### [readinessProbe.timeoutSeconds](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=readinessProbe.timeoutSeconds) + +**Default:** `1` + +### [replicaCount](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=replicaCount) + +**Default:** `1` + +### [resources](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=resources) + +**Default:** `{}` + +### [secret](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=secret) + +Create a new Kubernetes Secret for all sensitive configuration inputs. Each provided Secret is mounted automatically and made available to the Pod. If you want to use one or more existing Secrets, you can use the `extraEnvFrom` list to mount environment variables from string and secretMounts to mount files such as Certificates from Secrets. + +**Default:** + +``` +{"create":true,"enterprise":{},"kafka":{},"login":{"github":{},"google":{},"jwtSecret":"","oidc":{},"okta":{}},"redpanda":{"adminApi":{}}} +``` + +### [secret.kafka](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=secret.kafka) + +Kafka Secrets. + +**Default:** `{}` + +### [secretMounts](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=secretMounts) + +SecretMounts is an abstraction to make a Secret available in the container's filesystem. Under the hood it creates a volume and a volume mount for the Redpanda Console container. + +**Default:** `[]` + +### [securityContext.runAsNonRoot](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=securityContext.runAsNonRoot) + +**Default:** `true` + +### [service.annotations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=service.annotations) + +**Default:** `{}` + +### [service.port](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=service.port) + +**Default:** `8080` + +### [service.targetPort](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=service.targetPort) + +Override the value in `console.config.server.listenPort` if not `nil` + +**Default:** `nil` + +### [service.type](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=service.type) + +**Default:** `"ClusterIP"` + +### [serviceAccount.annotations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=serviceAccount.annotations) + +Annotations to add to the service account. + +**Default:** `{}` + +### [serviceAccount.automountServiceAccountToken](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=serviceAccount.automountServiceAccountToken) + +Specifies whether a service account should automount API-Credentials + +**Default:** `true` + +### [serviceAccount.create](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=serviceAccount.create) + +Specifies whether a service account should be created. + +**Default:** `true` + +### [serviceAccount.name](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=serviceAccount.name) + +The name of the service account to use. If not set and `serviceAccount.create` is `true`, a name is generated using the `console.fullname` template + +**Default:** `""` + +### [strategy](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=strategy) + +**Default:** `{}` + +### [tests.enabled](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=tests.enabled) + +**Default:** `true` + +### [tolerations](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=tolerations) + +**Default:** `[]` + +### [topologySpreadConstraints](https://artifacthub.io/packages/helm/redpanda-data/console?modal=values&path=topologySpreadConstraints) + +**Default:** `[]` + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/examples/console-enterprise.yaml b/charts/redpanda/redpanda/5.9.10/charts/console/examples/console-enterprise.yaml new file mode 100644 index 0000000000..dc3f29197d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/examples/console-enterprise.yaml @@ -0,0 +1,94 @@ +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +image: + tag: master-8fcce39 + +resources: + limits: + cpu: 1 + memory: 2Gi + requests: + cpu: 100m + memory: 512Mi + +console: + config: + kafka: + brokers: + - bootstrap.mybrokers.com:9092 + clientId: redpanda-console + sasl: + enabled: true + mechanism: SCRAM-SHA-256 + username: console + # password: set via Helm secret / Env variable + tls: + enabled: false + login: + google: + enabled: true + clientId: redacted.apps.googleusercontent.com + # clientSecret: set via Helm secret / Env variable + directory: + # serviceAccountFilepath: set via Helm secret / Env variable + targetPrincipal: admin@mycompany.com + enterprise: + rbac: + enabled: true + roleBindingsFilepath: /etc/console/configs/role-bindings.yaml + roleBindings: + - roleName: viewer + metadata: + # Metadata properties will be shown in the UI. You can omit it if you want to + name: Developers + subjects: + # You can specify all groups or users from different providers here which shall be bound to the same role + - kind: group + provider: Google + name: engineering@mycompany.com + - kind: user + provider: Google + name: singleuser@mycompany.com + - roleName: admin + metadata: + name: Admin + subjects: + - kind: user + provider: Google + name: adminperson@mycompany.com + +secret: + create: true + kafka: + saslPassword: "redacted" + enterprise: + license: "redacted" + login: + google: + clientSecret: "redacted" + groupsServiceAccount: | + { + "type": "service_account", + "project_id": "redacted", + "private_key_id": "redacted", + "private_key": "-----BEGIN PRIVATE KEY-----\nREDACTED\n-----END PRIVATE KEY-----\n", + "client_email": "redacted@projectid.iam.gserviceaccount.com", + "client_id": "redacted", + "auth_uri": "https://accounts.google.com/o/oauth2/auth", + "token_uri": "https://oauth2.googleapis.com/token", + "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs", + "client_x509_cert_url": "https://www.googleapis.com/robot/v1/metadata/x509/redacted.iam.gserviceaccount.com" + } diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/NOTES.txt b/charts/redpanda/redpanda/5.9.10/charts/console/templates/NOTES.txt new file mode 100644 index 0000000000..7541881fc9 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/NOTES.txt @@ -0,0 +1,20 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- $notes := (get ((include "console.Notes" (dict "a" (list .))) | fromJson) "r") -}} +{{- range $_, $note := $notes }} +{{ $note }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_chart.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_chart.go.tpl new file mode 100644 index 0000000000..47f236d6ff --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_chart.go.tpl @@ -0,0 +1,13 @@ +{{- /* Generated from "chart.go" */ -}} + +{{- define "console.render" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $manifests := (list (get (fromJson (include "console.ServiceAccount" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.Secret" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.ConfigMap" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.Service" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.Ingress" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.Deployment" (dict "a" (list $dot) ))) "r") (get (fromJson (include "console.HorizontalPodAutoscaler" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $manifests) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_configmap.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_configmap.go.tpl new file mode 100644 index 0000000000..14673b0249 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_configmap.go.tpl @@ -0,0 +1,25 @@ +{{- /* Generated from "configmap.go" */ -}} + +{{- define "console.ConfigMap" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.configmap.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $data := (dict "config.yaml" (printf "# from .Values.console.config\n%s\n" (tpl (toYaml $values.console.config) $dot)) ) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.console.roles) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $data "roles.yaml" (tpl (toYaml (dict "roles" $values.console.roles )) $dot)) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.console.roleBindings) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $data "role-bindings.yaml" (tpl (toYaml (dict "roleBindings" $values.console.roleBindings )) $dot)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "ConfigMap" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") )) "data" $data ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_deployment.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_deployment.go.tpl new file mode 100644 index 0000000000..67aaf598fe --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_deployment.go.tpl @@ -0,0 +1,133 @@ +{{- /* Generated from "deployment.go" */ -}} + +{{- define "console.ContainerPort" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $listenPort := ((8080 | int) | int) -}} +{{- if (ne (toJson $values.service.targetPort) "null") -}} +{{- $listenPort = $values.service.targetPort -}} +{{- end -}} +{{- $configListenPort := (dig "server" "listenPort" (coalesce nil) $values.console.config) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.asintegral" (dict "a" (list $configListenPort) ))) "r")) ))) "r") -}} +{{- $ok_2 := $tmp_tuple_1.T2 -}} +{{- $asInt_1 := ($tmp_tuple_1.T1 | int) -}} +{{- if $ok_2 -}} +{{- $_is_returning = true -}} +{{- (dict "r" ($asInt_1 | int)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $listenPort) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.Deployment" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.deployment.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $replicas := (coalesce nil) -}} +{{- if (not $values.autoscaling.enabled) -}} +{{- $replicas = ($values.replicaCount | int) -}} +{{- end -}} +{{- $initContainers := (coalesce nil) -}} +{{- if (not (empty $values.initContainers.extraInitContainers)) -}} +{{- $initContainers = (fromYamlArray (tpl $values.initContainers.extraInitContainers $dot)) -}} +{{- end -}} +{{- $volumeMounts := (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "configs" "mountPath" "/etc/console/configs" "readOnly" true ))) -}} +{{- if $values.secret.create -}} +{{- $volumeMounts = (concat (default (list ) $volumeMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "secrets" "mountPath" "/etc/console/secrets" "readOnly" true )))) -}} +{{- end -}} +{{- range $_, $mount := $values.secretMounts -}} +{{- $volumeMounts = (concat (default (list ) $volumeMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" $mount.name "mountPath" $mount.path "subPath" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $mount.subPath "") ))) "r") )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $volumeMounts = (concat (default (list ) $volumeMounts) (default (list ) $values.extraVolumeMounts)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "strategy" (dict ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "apps/v1" "kind" "Deployment" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "annotations" $values.annotations )) "spec" (mustMergeOverwrite (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "strategy" (dict ) ) (dict "replicas" $replicas "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "console.SelectorLabels" (dict "a" (list $dot) ))) "r") )) "strategy" $values.strategy "template" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "annotations" (merge (dict ) (dict "checksum/config" (sha256sum (toYaml (get (fromJson (include "console.ConfigMap" (dict "a" (list $dot) ))) "r"))) ) $values.podAnnotations) "labels" (merge (dict ) (get (fromJson (include "console.SelectorLabels" (dict "a" (list $dot) ))) "r") $values.podLabels) )) "spec" (mustMergeOverwrite (dict "containers" (coalesce nil) ) (dict "imagePullSecrets" $values.imagePullSecrets "serviceAccountName" (get (fromJson (include "console.ServiceAccountName" (dict "a" (list $dot) ))) "r") "automountServiceAccountToken" $values.automountServiceAccountToken "securityContext" $values.podSecurityContext "nodeSelector" $values.nodeSelector "affinity" $values.affinity "topologySpreadConstraints" $values.topologySpreadConstraints "priorityClassName" $values.priorityClassName "tolerations" $values.tolerations "volumes" (get (fromJson (include "console.consolePodVolumes" (dict "a" (list $dot) ))) "r") "initContainers" $initContainers "containers" (concat (default (list ) (list (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" $dot.Chart.Name "command" $values.deployment.command "args" (concat (default (list ) (list "--config.filepath=/etc/console/configs/config.yaml")) (default (list ) $values.deployment.extraArgs)) "securityContext" $values.securityContext "image" (get (fromJson (include "console.containerImage" (dict "a" (list $dot) ))) "r") "imagePullPolicy" $values.image.pullPolicy "ports" (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "http" "containerPort" ((get (fromJson (include "console.ContainerPort" (dict "a" (list $dot) ))) "r") | int) "protocol" "TCP" ))) "volumeMounts" $volumeMounts "livenessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "httpGet" (mustMergeOverwrite (dict "port" 0 ) (dict "path" "/admin/health" "port" "http" )) )) (dict "initialDelaySeconds" ($values.livenessProbe.initialDelaySeconds | int) "periodSeconds" ($values.livenessProbe.periodSeconds | int) "timeoutSeconds" ($values.livenessProbe.timeoutSeconds | int) "successThreshold" ($values.livenessProbe.successThreshold | int) "failureThreshold" ($values.livenessProbe.failureThreshold | int) )) "readinessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "httpGet" (mustMergeOverwrite (dict "port" 0 ) (dict "path" "/admin/health" "port" "http" )) )) (dict "initialDelaySeconds" ($values.readinessProbe.initialDelaySeconds | int) "periodSeconds" ($values.readinessProbe.periodSeconds | int) "timeoutSeconds" ($values.readinessProbe.timeoutSeconds | int) "successThreshold" ($values.readinessProbe.successThreshold | int) "failureThreshold" ($values.readinessProbe.failureThreshold | int) )) "resources" $values.resources "env" (get (fromJson (include "console.consoleContainerEnv" (dict "a" (list $dot) ))) "r") "envFrom" $values.extraEnvFrom )))) (default (list ) $values.extraContainers)) )) )) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.containerImage" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tag := $dot.Chart.AppVersion -}} +{{- if (not (empty $values.image.tag)) -}} +{{- $tag = $values.image.tag -}} +{{- end -}} +{{- $image := (printf "%s:%s" $values.image.repository $tag) -}} +{{- if (not (empty $values.image.registry)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s" $values.image.registry $image)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $image) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.consoleContainerEnv" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.secret.create) -}} +{{- $vars := $values.extraEnv -}} +{{- if (not (empty $values.enterprise.licenseSecretRef.name)) -}} +{{- $vars = (concat (default (list ) $values.extraEnv) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "LICENSE" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" $values.enterprise.licenseSecretRef.name )) (dict "key" (default "enterprise-license" $values.enterprise.licenseSecretRef.key) )) )) )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $vars) | toJson -}} +{{- break -}} +{{- end -}} +{{- $possibleVars := (list (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.saslPassword "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SASL_PASSWORD" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "kafka-sasl-password" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.protobufGitBasicAuthPassword "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_PROTOBUF_GIT_BASICAUTH_PASSWORD" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "kafka-protobuf-git-basicauth-password" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.awsMskIamSecretKey "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SASL_AWSMSKIAM_SECRETKEY" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "kafka-sasl-aws-msk-iam-secret-key" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.tlsCa "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_TLS_CAFILEPATH" "value" "/etc/console/secrets/kafka-tls-ca" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.tlsCert "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_TLS_CERTFILEPATH" "value" "/etc/console/secrets/kafka-tls-cert" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.tlsKey "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_TLS_KEYFILEPATH" "value" "/etc/console/secrets/kafka-tls-key" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.schemaRegistryTlsCa "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SCHEMAREGISTRY_TLS_CAFILEPATH" "value" "/etc/console/secrets/kafka-schemaregistry-tls-ca" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.schemaRegistryTlsCert "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SCHEMAREGISTRY_TLS_CERTFILEPATH" "value" "/etc/console/secrets/kafka-schemaregistry-tls-cert" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.schemaRegistryTlsKey "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SCHEMAREGISTRY_TLS_KEYFILEPATH" "value" "/etc/console/secrets/kafka-schemaregistry-tls-key" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.kafka.schemaRegistryPassword "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "KAFKA_SCHEMAREGISTRY_PASSWORD" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "kafka-schema-registry-password" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" true "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_JWTSECRET" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-jwt-secret" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.google.clientSecret "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_GOOGLE_CLIENTSECRET" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-google-oauth-client-secret" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.google.groupsServiceAccount "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_GOOGLE_DIRECTORY_SERVICEACCOUNTFILEPATH" "value" "/etc/console/secrets/login-google-groups-service-account.json" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.github.clientSecret "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_GITHUB_CLIENTSECRET" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-github-oauth-client-secret" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.github.personalAccessToken "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_GITHUB_DIRECTORY_PERSONALACCESSTOKEN" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-github-personal-access-token" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.okta.clientSecret "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_OKTA_CLIENTSECRET" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-okta-client-secret" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.okta.directoryApiToken "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_OKTA_DIRECTORY_APITOKEN" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-okta-directory-api-token" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.login.oidc.clientSecret "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LOGIN_OIDC_CLIENTSECRET" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "login-oidc-client-secret" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.enterprise.license "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "LICENSE" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "enterprise-license" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.redpanda.adminApi.password "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_ADMINAPI_PASSWORD" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict "key" "redpanda-admin-api-password" )) )) )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.redpanda.adminApi.tlsCa "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_ADMINAPI_TLS_CAFILEPATH" "value" "/etc/console/secrets/redpanda-admin-api-tls-ca" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.redpanda.adminApi.tlsKey "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_ADMINAPI_TLS_KEYFILEPATH" "value" "/etc/console/secrets/redpanda-admin-api-tls-key" )) )) (mustMergeOverwrite (dict "Value" (coalesce nil) "EnvVar" (dict "name" "" ) ) (dict "Value" $values.secret.redpanda.adminApi.tlsCert "EnvVar" (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_ADMINAPI_TLS_CERTFILEPATH" "value" "/etc/console/secrets/redpanda-admin-api-tls-cert" )) ))) -}} +{{- $vars := $values.extraEnv -}} +{{- range $_, $possible := $possibleVars -}} +{{- if (not (empty $possible.Value)) -}} +{{- $vars = (concat (default (list ) $vars) (list $possible.EnvVar)) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $vars) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.consolePodVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $volumes := (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) (dict )) )) (dict "name" "configs" ))) -}} +{{- if $values.secret.create -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) )) (dict "name" "secrets" )))) -}} +{{- end -}} +{{- range $_, $mount := $values.secretMounts -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" $mount.secretName "defaultMode" $mount.defaultMode )) )) (dict "name" $mount.name )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $volumes) (default (list ) $values.extraVolumes))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.go.tpl new file mode 100644 index 0000000000..05ad609654 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.go.tpl @@ -0,0 +1,82 @@ +{{- /* Generated from "helpers.go" */ -}} + +{{- define "console.Name" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $name := (default $dot.Chart.Name $values.nameOverride) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.cleanForK8s" (dict "a" (list $name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.Fullname" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (ne $values.fullnameOverride "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.cleanForK8s" (dict "a" (list $values.fullnameOverride) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $name := (default $dot.Chart.Name $values.nameOverride) -}} +{{- if (contains $name $dot.Release.Name) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.cleanForK8s" (dict "a" (list $dot.Release.Name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.cleanForK8s" (dict "a" (list (printf "%s-%s" $dot.Release.Name $name)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.ChartLabel" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $chart := (printf "%s-%s" $dot.Chart.Name $dot.Chart.Version) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.cleanForK8s" (dict "a" (list (replace "+" "_" $chart)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.Labels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $labels := (dict "helm.sh/chart" (get (fromJson (include "console.ChartLabel" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/managed-by" $dot.Release.Service ) -}} +{{- if (ne $dot.Chart.AppVersion "") -}} +{{- $_ := (set $labels "app.kubernetes.io/version" $dot.Chart.AppVersion) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $labels (get (fromJson (include "console.SelectorLabels" (dict "a" (list $dot) ))) "r") $values.commonLabels)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.SelectorLabels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "app.kubernetes.io/name" (get (fromJson (include "console.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.cleanForK8s" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (trimSuffix "-" (trunc (63 | int) $s))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.tpl new file mode 100644 index 0000000000..ee2ab5d9b8 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_helpers.tpl @@ -0,0 +1,25 @@ +{{/* +Expand the name of the chart. +Used by tests/test-connection.yaml +*/}} +{{- define "console.name" -}} +{{- get ((include "console.Name" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If release name contains chart name it will be used as a full name. +Used by tests/test-connection.yaml +*/}} +{{- define "console.fullname" -}} +{{- get ((include "console.Fullname" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Common labels +Used by tests/test-connection.yaml +*/}} +{{- define "console.labels" -}} +{{- (get ((include "console.Labels" (dict "a" (list .))) | fromJson) "r") | toYaml -}} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_hpa.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_hpa.go.tpl new file mode 100644 index 0000000000..5c3b33beda --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_hpa.go.tpl @@ -0,0 +1,25 @@ +{{- /* Generated from "hpa.go" */ -}} + +{{- define "console.HorizontalPodAutoscaler" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.autoscaling.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $metrics := (list ) -}} +{{- if (ne (toJson $values.autoscaling.targetCPUUtilizationPercentage) "null") -}} +{{- $metrics = (concat (default (list ) $metrics) (list (mustMergeOverwrite (dict "type" "" ) (dict "type" "Resource" "resource" (mustMergeOverwrite (dict "name" "" "target" (dict "type" "" ) ) (dict "name" "cpu" "target" (mustMergeOverwrite (dict "type" "" ) (dict "type" "Utilization" "averageUtilization" $values.autoscaling.targetCPUUtilizationPercentage )) )) )))) -}} +{{- end -}} +{{- if (ne (toJson $values.autoscaling.targetMemoryUtilizationPercentage) "null") -}} +{{- $metrics = (concat (default (list ) $metrics) (list (mustMergeOverwrite (dict "type" "" ) (dict "type" "Resource" "resource" (mustMergeOverwrite (dict "name" "" "target" (dict "type" "" ) ) (dict "name" "memory" "target" (mustMergeOverwrite (dict "type" "" ) (dict "type" "Utilization" "averageUtilization" $values.autoscaling.targetMemoryUtilizationPercentage )) )) )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "scaleTargetRef" (dict "kind" "" "name" "" ) "maxReplicas" 0 ) "status" (dict "desiredReplicas" 0 "currentMetrics" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "autoscaling/v2" "kind" "HorizontalPodAutoscaler" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict "scaleTargetRef" (dict "kind" "" "name" "" ) "maxReplicas" 0 ) (dict "scaleTargetRef" (mustMergeOverwrite (dict "kind" "" "name" "" ) (dict "apiVersion" "apps/v1" "kind" "Deployment" "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") )) "minReplicas" ($values.autoscaling.minReplicas | int) "maxReplicas" ($values.autoscaling.maxReplicas | int) "metrics" $metrics )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_ingress.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_ingress.go.tpl new file mode 100644 index 0000000000..0df05e870b --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_ingress.go.tpl @@ -0,0 +1,46 @@ +{{- /* Generated from "ingress.go" */ -}} + +{{- define "console.Ingress" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.ingress.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $tls := (coalesce nil) -}} +{{- range $_, $t := $values.ingress.tls -}} +{{- $hosts := (coalesce nil) -}} +{{- range $_, $host := $t.hosts -}} +{{- $hosts = (concat (default (list ) $hosts) (list (tpl $host $dot))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $tls = (concat (default (list ) $tls) (list (mustMergeOverwrite (dict ) (dict "secretName" $t.secretName "hosts" $hosts )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $rules := (coalesce nil) -}} +{{- range $_, $host := $values.ingress.hosts -}} +{{- $paths := (coalesce nil) -}} +{{- range $_, $path := $host.paths -}} +{{- $paths = (concat (default (list ) $paths) (list (mustMergeOverwrite (dict "pathType" (coalesce nil) "backend" (dict ) ) (dict "path" $path.path "pathType" $path.pathType "backend" (mustMergeOverwrite (dict ) (dict "service" (mustMergeOverwrite (dict "name" "" "port" (dict ) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "port" (mustMergeOverwrite (dict ) (dict "number" ($values.service.port | int) )) )) )) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $rules = (concat (default (list ) $rules) (list (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "http" (mustMergeOverwrite (dict "paths" (coalesce nil) ) (dict "paths" $paths )) )) (dict "host" (tpl $host.host $dot) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "kind" "Ingress" "apiVersion" "networking.k8s.io/v1" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") "annotations" $values.ingress.annotations )) "spec" (mustMergeOverwrite (dict ) (dict "ingressClassName" $values.ingress.className "tls" $tls "rules" $rules )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_notes.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_notes.go.tpl new file mode 100644 index 0000000000..6b58b21ef4 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_notes.go.tpl @@ -0,0 +1,40 @@ +{{- /* Generated from "notes.go" */ -}} + +{{- define "console.Notes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $commands := (list `1. Get the application URL by running these commands:`) -}} +{{- if $values.ingress.enabled -}} +{{- $scheme := "http" -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.ingress.tls) ))) "r") | int) (0 | int)) -}} +{{- $scheme = "https" -}} +{{- end -}} +{{- range $_, $host := $values.ingress.hosts -}} +{{- range $_, $path := $host.paths -}} +{{- $commands = (concat (default (list ) $commands) (list (printf "%s://%s%s" $scheme $host.host $path.path))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- else -}}{{- if (contains "NodePort" (toString $values.service.type)) -}} +{{- $commands = (concat (default (list ) $commands) (list (printf ` export NODE_PORT=$(kubectl get --namespace %s -o jsonpath="{.spec.ports[0].nodePort}" services %s)` $dot.Release.Namespace (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r")) (printf ` export NODE_IP=$(kubectl get nodes --namespace %s -o jsonpath="{.items[0].status.addresses[0].address}")` $dot.Release.Namespace) " echo http://$NODE_IP:$NODE_PORT")) -}} +{{- else -}}{{- if (contains "NodePort" (toString $values.service.type)) -}} +{{- $commands = (concat (default (list ) $commands) (list ` NOTE: It may take a few minutes for the LoadBalancer IP to be available.` (printf ` You can watch the status of by running 'kubectl get --namespace %s svc -w %s'` $dot.Release.Namespace (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r")) (printf ` export SERVICE_IP=$(kubectl get svc --namespace %s %s --template "{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}")` $dot.Release.Namespace (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r")) (printf ` echo http://$SERVICE_IP:%d` ($values.service.port | int)))) -}} +{{- else -}}{{- if (contains "ClusterIP" (toString $values.service.type)) -}} +{{- $commands = (concat (default (list ) $commands) (list (printf ` export POD_NAME=$(kubectl get pods --namespace %s -l "app.kubernetes.io/name=%s,app.kubernetes.io/instance=%s" -o jsonpath="{.items[0].metadata.name}")` $dot.Release.Namespace (get (fromJson (include "console.Name" (dict "a" (list $dot) ))) "r") $dot.Release.Name) (printf ` export CONTAINER_PORT=$(kubectl get pod --namespace %s $POD_NAME -o jsonpath="{.spec.containers[0].ports[0].containerPort}")` $dot.Release.Namespace) ` echo "Visit http://127.0.0.1:8080 to use your application"` (printf ` kubectl --namespace %s port-forward $POD_NAME 8080:$CONTAINER_PORT` $dot.Release.Namespace))) -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $commands) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_secret.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_secret.go.tpl new file mode 100644 index 0000000000..6af16b1c83 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_secret.go.tpl @@ -0,0 +1,22 @@ +{{- /* Generated from "secret.go" */ -}} + +{{- define "console.Secret" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.secret.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $jwtSecret := $values.secret.login.jwtSecret -}} +{{- if (eq $jwtSecret "") -}} +{{- $jwtSecret = (randAlphaNum (32 | int)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict "kafka-sasl-password" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.saslPassword "") ))) "r") "kafka-protobuf-git-basicauth-password" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.protobufGitBasicAuthPassword "") ))) "r") "kafka-sasl-aws-msk-iam-secret-key" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.awsMskIamSecretKey "") ))) "r") "kafka-tls-ca" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.tlsCa "") ))) "r") "kafka-tls-cert" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.tlsCert "") ))) "r") "kafka-tls-key" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.tlsKey "") ))) "r") "kafka-schema-registry-password" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.schemaRegistryPassword "") ))) "r") "kafka-schemaregistry-tls-ca" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.schemaRegistryTlsCa "") ))) "r") "kafka-schemaregistry-tls-cert" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.schemaRegistryTlsCert "") ))) "r") "kafka-schemaregistry-tls-key" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.kafka.schemaRegistryTlsKey "") ))) "r") "login-jwt-secret" $jwtSecret "login-google-oauth-client-secret" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.google.clientSecret "") ))) "r") "login-google-groups-service-account.json" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.google.groupsServiceAccount "") ))) "r") "login-github-oauth-client-secret" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.github.clientSecret "") ))) "r") "login-github-personal-access-token" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.github.personalAccessToken "") ))) "r") "login-okta-client-secret" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.okta.clientSecret "") ))) "r") "login-okta-directory-api-token" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.okta.directoryApiToken "") ))) "r") "login-oidc-client-secret" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.login.oidc.clientSecret "") ))) "r") "enterprise-license" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.enterprise.license "") ))) "r") "redpanda-admin-api-password" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.redpanda.adminApi.password "") ))) "r") "redpanda-admin-api-tls-ca" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.redpanda.adminApi.tlsCa "") ))) "r") "redpanda-admin-api-tls-cert" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.redpanda.adminApi.tlsCert "") ))) "r") "redpanda-admin-api-tls-key" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.secret.redpanda.adminApi.tlsKey "") ))) "r") ) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_service.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_service.go.tpl new file mode 100644 index 0000000000..8fac3d4542 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_service.go.tpl @@ -0,0 +1,20 @@ +{{- /* Generated from "service.go" */ -}} + +{{- define "console.Service" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $port := (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "http" "port" (($values.service.port | int) | int) "protocol" "TCP" )) -}} +{{- if (ne (toJson $values.service.targetPort) "null") -}} +{{- $_ := (set $port "targetPort" $values.service.targetPort) -}} +{{- end -}} +{{- if (and (contains "NodePort" (toString $values.service.type)) (ne (toJson $values.service.nodePort) "null")) -}} +{{- $_ := (set $port "nodePort" $values.service.nodePort) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Service" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") "annotations" $values.service.annotations )) "spec" (mustMergeOverwrite (dict ) (dict "type" $values.service.type "selector" (get (fromJson (include "console.SelectorLabels" (dict "a" (list $dot) ))) "r") "ports" (list $port) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_serviceaccount.go.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_serviceaccount.go.tpl new file mode 100644 index 0000000000..5a49ba3fdb --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_serviceaccount.go.tpl @@ -0,0 +1,39 @@ +{{- /* Generated from "serviceaccount.go" */ -}} + +{{- define "console.ServiceAccountName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if $values.serviceAccount.create -}} +{{- if (ne $values.serviceAccount.name "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $values.serviceAccount.name) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "console.Fullname" (dict "a" (list $dot) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default "default" $values.serviceAccount.name)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "console.ServiceAccount" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.serviceAccount.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "kind" "ServiceAccount" "apiVersion" "v1" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "console.ServiceAccountName" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "console.Labels" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "annotations" $values.serviceAccount.annotations )) "automountServiceAccountToken" $values.serviceAccount.automountServiceAccountToken ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/_shims.tpl b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_shims.tpl new file mode 100644 index 0000000000..1e6d0425c3 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/_shims.tpl @@ -0,0 +1,289 @@ +{{- /* Generated from "bootstrap.go" */ -}} + +{{- define "_shims.typetest" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs $typ $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.typeassertion" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not (typeIs $typ $value)) -}} +{{- $_ := (fail (printf "expected type of %q got: %T" $typ $value)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $value) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.dicttest" -}} +{{- $m := (index .a 0) -}} +{{- $key := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (hasKey $m $key) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (index $m $key) true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.compact" -}} +{{- $args := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $out := (dict ) -}} +{{- range $i, $e := $args -}} +{{- $_ := (set $out (printf "T%d" ((add (1 | int) $i) | int)) $e) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $out) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.deref" -}} +{{- $ptr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $ptr) "null") -}} +{{- $_ := (fail "nil dereference") -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.len" -}} +{{- $m := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $m) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (0 | int)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (len $m)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Deref" -}} +{{- $ptr := (index .a 0) -}} +{{- $def := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $ptr) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $def) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Equal" -}} +{{- $a := (index .a 0) -}} +{{- $b := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (and (eq (toJson $a) "null") (eq (toJson $b) "null")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (eq $a $b)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.lookup" -}} +{{- $apiVersion := (index .a 0) -}} +{{- $kind := (index .a 1) -}} +{{- $namespace := (index .a 2) -}} +{{- $name := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (lookup $apiVersion $kind $namespace $name) -}} +{{- if (empty $result) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (coalesce nil) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $result true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asnumeric" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asintegral" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (or (typeIs "int64" $value) (typeIs "int" $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (and (typeIs "float64" $value) (eq (floor $value) $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.parseResource" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $repr) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (float64 $repr) 1.0)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (not (typeIs "string" $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity expected string or float64 got: %T (%v)" $repr $repr)) -}} +{{- end -}} +{{- if (not (regexMatch `^[0-9]+(\.[0-9]{0,6})?(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$` $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity: %q" $repr)) -}} +{{- end -}} +{{- $reprStr := (toString $repr) -}} +{{- $unit := (regexFind "(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)$" $repr) -}} +{{- $numeric := (float64 (substr (0 | int) ((sub ((get (fromJson (include "_shims.len" (dict "a" (list $reprStr) ))) "r") | int) ((get (fromJson (include "_shims.len" (dict "a" (list $unit) ))) "r") | int)) | int) $reprStr)) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list (dict "" 1.0 "m" 0.001 "k" (1000 | int) "M" (1000000 | int) "G" (1000000000 | int) "T" (1000000000000 | int) "P" (1000000000000000 | int) "Ki" (1024 | int) "Mi" (1048576 | int) "Gi" (1073741824 | int) "Ti" (1099511627776 | int) "Pi" (1125899906842624 | int) ) $unit (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_1.T2 -}} +{{- $scale := ($tmp_tuple_1.T1 | float64) -}} +{{- if (not $ok) -}} +{{- $_ := (fail (printf "unknown unit: %q" $unit)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $numeric $scale)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MustParse" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_2.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_2.T1 | float64) -}} +{{- $strs := (list "" "m" "k" "M" "G" "T" "P" "Ki" "Mi" "Gi" "Ti" "Pi") -}} +{{- $scales := (list 1.0 0.001 (1000 | int) (1000000 | int) (1000000000 | int) (1000000000000 | int) (1000000000000000 | int) (1024 | int) (1048576 | int) (1073741824 | int) (1099511627776 | int) (1125899906842624 | int)) -}} +{{- $idx := -1 -}} +{{- range $i, $s := $scales -}} +{{- if (eq ($s | float64) ($scale | float64)) -}} +{{- $idx = $i -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (eq $idx -1) -}} +{{- $_ := (fail (printf "unknown scale: %v" $scale)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s%s" (toString $numeric) (index $strs $idx))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_Value" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_3.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_3.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf $numeric $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MilliValue" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_4.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_4.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf ((mulf $numeric 1000.0) | float64) $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.render-manifest" -}} +{{- $tpl := (index . 0) -}} +{{- $dot := (index . 1) -}} +{{- $manifests := (get ((include $tpl (dict "a" (list $dot))) | fromJson) "r") -}} +{{- if not (typeIs "[]interface {}" $manifests) -}} +{{- $manifests = (list $manifests) -}} +{{- end -}} +{{- range $_, $manifest := $manifests -}} +{{- if ne (toJson $manifest) "null" }} +--- +{{toYaml (unset (unset $manifest "status") "creationTimestamp")}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/entry-point.yaml b/charts/redpanda/redpanda/5.9.10/charts/console/templates/entry-point.yaml new file mode 100644 index 0000000000..01fb6d68b2 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/entry-point.yaml @@ -0,0 +1,17 @@ +{{- /* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "console.render" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/templates/tests/test-connection.yaml b/charts/redpanda/redpanda/5.9.10/charts/console/templates/tests/test-connection.yaml new file mode 100644 index 0000000000..de17fb2b1d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/templates/tests/test-connection.yaml @@ -0,0 +1,22 @@ +{{- if .Values.tests.enabled }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "console.fullname" . }}-test-connection" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- include "console.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": test +spec: +{{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} +{{- end }} + containers: + - name: wget + image: busybox + command: ['wget'] + args: ['{{ include "console.fullname" . }}:{{ .Values.service.port }}'] + restartPolicy: Never + priorityClassName: {{ .Values.priorityClassName }} +{{- end }} \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/values.schema.json b/charts/redpanda/redpanda/5.9.10/charts/console/values.schema.json new file mode 100644 index 0000000000..f4f369e98a --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/values.schema.json @@ -0,0 +1,323 @@ +{ + "$schema": "http://json-schema.org/schema#", + "type": "object", + "required": [ + "image" + ], + "properties": { + "affinity": { + "type": "object" + }, + "autoscaling": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "maxReplicas": { + "type": "integer" + }, + "minReplicas": { + "type": "integer" + }, + "targetCPUUtilizationPercentage": { + "type": "integer" + } + } + }, + "configmap": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + } + } + }, + "console": { + "type": "object" + }, + "deployment": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + } + } + }, + "extraContainers": { + "type": "array" + }, + "extraEnv": { + "type": "array" + }, + "extraEnvFrom": { + "type": "array" + }, + "extraVolumeMounts": { + "type": "array" + }, + "extraVolumes": { + "type": "array" + }, + "fullnameOverride": { + "type": "string" + }, + "image": { + "type": "object", + "required": [ + "repository" + ], + "properties": { + "pullPolicy": { + "type": "string" + }, + "registry": { + "type": "string" + }, + "repository": { + "type": "string", + "minLength": 1 + }, + "tag": { + "type": "string" + } + } + }, + "imagePullSecrets": { + "type": "array" + }, + "ingress": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "className": { + "type": ["string", "null"] + }, + "enabled": { + "type": "boolean" + }, + "hosts": { + "type": "array", + "items": { + "type": "object", + "properties": { + "host": { + "type": "string" + }, + "paths": { + "type": "array", + "items": { + "type": "object", + "properties": { + "path": { + "type": "string" + }, + "pathType": { + "type": "string" + } + } + } + } + } + } + }, + "tls": { + "type": "array" + } + } + }, + "livenessProbe": { + "type": "object", + "properties": { + "failureThreshold": { + "type": "integer" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + } + }, + "nameOverride": { + "type": "string" + }, + "nodeSelector": { + "type": "object" + }, + "annotations": { + "type": "object" + }, + "podAnnotations": { + "type": "object" + }, + "podSecurityContext": { + "type": "object", + "properties": { + "fsGroup": { + "type": "integer" + }, + "runAsUser": { + "type": "integer" + } + } + }, + "readinessProbe": { + "type": "object", + "properties": { + "failureThreshold": { + "type": "integer" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + } + }, + "replicaCount": { + "type": "integer" + }, + "resources": { + "type": "object" + }, + "secret": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "enterprise": { + "type": "object" + }, + "kafka": { + "type": "object" + }, + "login": { + "type": "object", + "properties": { + "jwtSecret": { + "type": "string" + }, + "github": { + "type": "object" + }, + "google": { + "type": "object" + }, + "oidc": { + "type": "object" + }, + "okta": { + "type": "object" + } + } + }, + "redpanda": { + "type": "object", + "properties": { + "adminApi": { + "type": "object" + } + } + } + } + }, + "secretMounts": { + "type": "array" + }, + "securityContext": { + "type": "object", + "properties": { + "runAsNonRoot": { + "type": "boolean" + } + } + }, + "service": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "port": { + "type": "integer" + }, + "nodePort": { + "type": "integer" + }, + "targetPort": { + "anyOf": [ + { + "type": "integer" + }, + { + "type": "null" + } + ] + }, + "type": { + "type": "string" + } + } + }, + "automountServiceAccountToken": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "create": { + "type": "boolean" + }, + "automountServiceAccountToken": { + "type": "boolean" + }, + "name": { + "type": "string" + } + } + }, + "tolerations": { + "type": "array" + }, + "initContainers": { + "type": "object", + "properties": { + "extraInitContainers": { + "type": "string" + } + } + }, + "strategy": { + "type": "object" + }, + "tests": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + } + } + } +} diff --git a/charts/redpanda/redpanda/5.9.10/charts/console/values.yaml b/charts/redpanda/redpanda/5.9.10/charts/console/values.yaml new file mode 100644 index 0000000000..4825fc4876 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/charts/console/values.yaml @@ -0,0 +1,279 @@ +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# Default values for console. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +replicaCount: 1 + +# -- Redpanda Console Docker image settings. +image: + registry: docker.redpanda.com + # -- Docker repository from which to pull the Redpanda Docker image. + repository: redpandadata/console + # -- The imagePullPolicy. + pullPolicy: IfNotPresent + # -- The Redpanda Console version. + # See DockerHub for: + # [All stable versions](https://hub.docker.com/r/redpandadata/console/tags) + # and [all unstable versions](https://hub.docker.com/r/redpandadata/console-unstable/tags). + # @default -- `Chart.appVersion` + tag: "" + +# -- Pull secrets may be used to provide credentials to image repositories +# See https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ +imagePullSecrets: [] + +# -- Override `console.name` template. +nameOverride: "" +# -- Override `console.fullname` template. +fullnameOverride: "" + +# -- Automount API credentials for the Service Account into the pod. +automountServiceAccountToken: true + +serviceAccount: + # -- Specifies whether a service account should be created. + create: true + # -- Specifies whether a service account should automount API-Credentials + automountServiceAccountToken: true + # -- Annotations to add to the service account. + annotations: {} + # -- The name of the service account to use. + # If not set and `serviceAccount.create` is `true`, + # a name is generated using the `console.fullname` template + name: "" + +# Common labels to add to all the pods +commonLabels: {} + +# -- Annotations to add to the deployment. +annotations: {} + +podAnnotations: {} + +podLabels: {} + +podSecurityContext: + runAsUser: 99 + fsGroup: 99 + +securityContext: + runAsNonRoot: true + # capabilities: + # drop: + # - ALL + # readOnlyRootFilesystem: true + # runAsNonRoot: true + # runAsUser: 1000 + +service: + type: ClusterIP + port: 8080 + # nodePort: 30001 + # -- Override the value in `console.config.server.listenPort` if not `nil` + targetPort: + annotations: {} + +ingress: + enabled: false + className: + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + hosts: + - host: chart-example.local + paths: + - path: / + pathType: ImplementationSpecific + tls: [] + # - secretName: chart-example-tls + # hosts: + # - chart-example.local + +resources: {} + # We usually recommend not to specify default resources and to leave this as a conscious + # choice for the user. This also increases chances charts run on environments with little + # resources, such as minikube. If you want to specify resources, uncomment the following + # lines, adjust them as necessary, and remove the curly braces after 'resources:'. + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 100 + targetCPUUtilizationPercentage: 80 + # targetMemoryUtilizationPercentage: 80 + +nodeSelector: {} + +tolerations: [] + +affinity: {} + +topologySpreadConstraints: [] + +# -- PriorityClassName given to Pods. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). +priorityClassName: "" + +console: + # -- Settings for the `Config.yaml` (required). + # For a reference of configuration settings, + # see the [Redpanda Console documentation](https://docs.redpanda.com/docs/reference/console/config/). + config: {} + # roles: + # roleBindings: + +# -- Additional environment variables for the Redpanda Console Deployment. +extraEnv: [] + # - name: KAFKA_RACKID + # value: "1" + +# -- Additional environment variables for Redpanda Console mapped from Secret or ConfigMap. +extraEnvFrom: [] +# - secretRef: +# name: kowl-config-secret + +# -- Add additional volumes, such as for TLS keys. +extraVolumes: [] +# - name: kafka-certs +# secret: +# secretName: kafka-certs +# - name: config +# configMap: +# name: console-config + +# -- Add additional volume mounts, such as for TLS keys. +extraVolumeMounts: [] +# - name: kafka-certs # Must match the volume name +# mountPath: /etc/kafka/certs +# readOnly: true + +# -- Add additional containers, such as for oauth2-proxy. +extraContainers: [] + +# -- Any initContainers defined should be written here +initContainers: + # -- Additional set of init containers + extraInitContainers: |- +# - name: "test-init-container" +# image: "mintel/docker-alpine-bash-curl-jq:latest" +# command: [ "/bin/bash", "-c" ] +# args: +# - | +# set -xe +# echo "Hello World!" + +# -- SecretMounts is an abstraction to make a Secret available in the container's filesystem. +# Under the hood it creates a volume and a volume mount for the Redpanda Console container. +secretMounts: [] +# - name: kafka-certs +# secretName: kafka-certs +# path: /etc/console/certs +# defaultMode: 0755 + +# -- Create a new Kubernetes Secret for all sensitive configuration inputs. +# Each provided Secret is mounted automatically and made available to the +# Pod. +# If you want to use one or more existing Secrets, +# you can use the `extraEnvFrom` list to mount environment variables from string and secretMounts to mount files such as Certificates from Secrets. +secret: + create: true + + # Secret values in case you want the chart to create a Secret. All Certificates are mounted + # as files and the path to those files are configured through environment variables so + # that Console can automatically pick them up. + # -- Kafka Secrets. + kafka: {} + # saslPassword: + # awsMskIamSecretKey: + # tlsCa: + # tlsCert: + # tlsKey: + # tlsPassphrase: + # schemaRegistryPassword: + # schemaRegistryTlsCa: + # schemaRegistryTlsCert: + # schemaRegistryTlsKey: + # protobufGitBasicAuthPassword + # Enterprise version secrets + # - SSO secrets (Enterprise version). + login: + # Configurable JWT value + jwtSecret: "" + google: {} + # clientSecret: + # groupsServiceAccount: + github: {} + # clientSecret: + # personalAccessToken: + okta: {} + # clientSecret: + # directoryApiToken: + oidc: {} + # clientSecret: + + enterprise: {} + # license: + + redpanda: + adminApi: {} + # password: + # tlsCa: + # tlsCert: + # tlsKey: + +# -- Settings for license key, as an alternative to secret.enterprise when +# a license secret is available +enterprise: + licenseSecretRef: + name: "" + key: "" + +# -- Settings for liveness and readiness probes. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes). +livenessProbe: + # initialDelaySeconds: 0 + periodSeconds: 10 + timeoutSeconds: 1 + successThreshold: 1 + failureThreshold: 3 + +readinessProbe: + # -- Grant time to test connectivity to upstream services such as Kafka and Schema Registry. + initialDelaySeconds: 10 + periodSeconds: 10 + timeoutSeconds: 1 + successThreshold: 1 + failureThreshold: 3 + +configmap: + create: true +deployment: + create: true + +strategy: {} + +tests: + enabled: true diff --git a/charts/redpanda/redpanda/5.9.10/templates/NOTES.txt b/charts/redpanda/redpanda/5.9.10/templates/NOTES.txt new file mode 100644 index 0000000000..6992f8e36d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/NOTES.txt @@ -0,0 +1,26 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} + +{{- $warnings := (get ((include "redpanda.Warnings" (dict "a" (list .))) | fromJson) "r") }} +{{- range $_, $warning := $warnings }} +{{ $warning }} +{{- end }} + +{{- $notes := (get ((include "redpanda.Notes" (dict "a" (list .))) | fromJson) "r") }} +{{- range $_, $note := $notes }} +{{ $note }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/_cert-issuers.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_cert-issuers.go.tpl new file mode 100644 index 0000000000..acfd4c46c3 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_cert-issuers.go.tpl @@ -0,0 +1,57 @@ +{{- /* Generated from "cert_issuers.go" */ -}} + +{{- define "redpanda.CertIssuers" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "redpanda.certIssuersAndCAs" (dict "a" (list $dot) ))) "r")) ))) "r") -}} +{{- $issuers := $tmp_tuple_1.T1 -}} +{{- $_is_returning = true -}} +{{- (dict "r" $issuers) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RootCAs" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "redpanda.certIssuersAndCAs" (dict "a" (list $dot) ))) "r")) ))) "r") -}} +{{- $cas := $tmp_tuple_2.T2 -}} +{{- $_is_returning = true -}} +{{- (dict "r" $cas) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.certIssuersAndCAs" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $issuers := (coalesce nil) -}} +{{- $certs := (coalesce nil) -}} +{{- if (not (get (fromJson (include "redpanda.TLSEnabled" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $issuers $certs)) | toJson -}} +{{- break -}} +{{- end -}} +{{- range $name, $data := $values.tls.certs -}} +{{- if (or (not (empty $data.secretRef)) (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $data.enabled true) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- if (eq (toJson $data.issuerRef) "null") -}} +{{- $issuers = (concat (default (list ) $issuers) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "cert-manager.io/v1" "kind" "Issuer" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf `%s-%s-selfsigned-issuer` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "selfSigned" (mustMergeOverwrite (dict ) (dict )) )) (dict )) )))) -}} +{{- end -}} +{{- $issuers = (concat (default (list ) $issuers) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "cert-manager.io/v1" "kind" "Issuer" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf `%s-%s-root-issuer` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "ca" (mustMergeOverwrite (dict "secretName" "" ) (dict "secretName" (printf `%s-%s-root-certificate` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) )) )) (dict )) )))) -}} +{{- $certs = (concat (default (list ) $certs) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "secretName" "" "issuerRef" (dict "name" "" ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "cert-manager.io/v1" "kind" "Certificate" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf `%s-%s-root-certificate` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict "secretName" "" "issuerRef" (dict "name" "" ) ) (dict "duration" (get (fromJson (include "_shims.time_Duration_String" (dict "a" (list (get (fromJson (include "_shims.time_ParseDuration" (dict "a" (list (default "43800h" $data.duration)) ))) "r")) ))) "r") "isCA" true "commonName" (printf `%s-%s-root-certificate` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "secretName" (printf `%s-%s-root-certificate` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "privateKey" (mustMergeOverwrite (dict ) (dict "algorithm" "ECDSA" "size" (256 | int) )) "issuerRef" (mustMergeOverwrite (dict "name" "" ) (dict "name" (printf `%s-%s-selfsigned-issuer` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $name) "kind" "Issuer" "group" "cert-manager.io" )) )) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $issuers $certs)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_certs.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_certs.go.tpl new file mode 100644 index 0000000000..b87a68b949 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_certs.go.tpl @@ -0,0 +1,71 @@ +{{- /* Generated from "certs.go" */ -}} + +{{- define "redpanda.ClientCerts" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not (get (fromJson (include "redpanda.TLSEnabled" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $values := $dot.Values.AsMap -}} +{{- $fullname := (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") -}} +{{- $service := (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r") -}} +{{- $ns := $dot.Release.Namespace -}} +{{- $domain := (trimSuffix "." $values.clusterDomain) -}} +{{- $certs := (coalesce nil) -}} +{{- range $name, $data := $values.tls.certs -}} +{{- if (or (not (empty $data.secretRef)) (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $data.enabled true) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- $names := (coalesce nil) -}} +{{- if (or (eq (toJson $data.issuerRef) "null") (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $data.applyInternalDNSNames false) ))) "r")) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s-cluster.%s.%s.svc.%s" $fullname $service $ns $domain))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s-cluster.%s.%s.svc" $fullname $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s-cluster.%s.%s" $fullname $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s-cluster.%s.%s.svc.%s" $fullname $service $ns $domain))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s-cluster.%s.%s.svc" $fullname $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s-cluster.%s.%s" $fullname $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s.%s.svc.%s" $service $ns $domain))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s.%s.svc" $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "%s.%s" $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s.%s.svc.%s" $service $ns $domain))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s.%s.svc" $service $ns))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s.%s" $service $ns))) -}} +{{- end -}} +{{- if (ne (toJson $values.external.domain) "null") -}} +{{- $names = (concat (default (list ) $names) (list (tpl $values.external.domain $dot))) -}} +{{- $names = (concat (default (list ) $names) (list (printf "*.%s" (tpl $values.external.domain $dot)))) -}} +{{- end -}} +{{- $duration := (default "43800h" $data.duration) -}} +{{- $issuerRef := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $data.issuerRef (mustMergeOverwrite (dict "name" "" ) (dict "kind" "Issuer" "group" "cert-manager.io" "name" (printf "%s-%s-root-issuer" $fullname $name) ))) ))) "r") -}} +{{- $certs = (concat (default (list ) $certs) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "secretName" "" "issuerRef" (dict "name" "" ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "cert-manager.io/v1" "kind" "Certificate" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-%s-cert" $fullname $name) "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace )) "spec" (mustMergeOverwrite (dict "secretName" "" "issuerRef" (dict "name" "" ) ) (dict "dnsNames" $names "duration" (get (fromJson (include "_shims.time_Duration_String" (dict "a" (list (get (fromJson (include "_shims.time_ParseDuration" (dict "a" (list $duration) ))) "r")) ))) "r") "isCA" false "issuerRef" $issuerRef "secretName" (printf "%s-%s-cert" $fullname $name) "privateKey" (mustMergeOverwrite (dict ) (dict "algorithm" "ECDSA" "size" (256 | int) )) )) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $name := $values.listeners.kafka.tls.cert -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $values.tls.certs $name (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_1.T2 -}} +{{- $data := $tmp_tuple_1.T1 -}} +{{- if (not $ok) -}} +{{- $_ := (fail (printf "Certificate %q referenced but not defined" $name)) -}} +{{- end -}} +{{- if (or (not (empty $data.secretRef)) (not (get (fromJson (include "redpanda.ClientAuthRequired" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $certs) | toJson -}} +{{- break -}} +{{- end -}} +{{- $issuerRef := (mustMergeOverwrite (dict "name" "" ) (dict "group" "cert-manager.io" "kind" "Issuer" "name" (printf "%s-%s-root-issuer" $fullname $name) )) -}} +{{- if (ne (toJson $data.issuerRef) "null") -}} +{{- $issuerRef = $data.issuerRef -}} +{{- $_ := (set $issuerRef "group" "cert-manager.io") -}} +{{- end -}} +{{- $duration := (default "43800h" $data.duration) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $certs) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "secretName" "" "issuerRef" (dict "name" "" ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "cert-manager.io/v1" "kind" "Certificate" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-client" $fullname) "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict "secretName" "" "issuerRef" (dict "name" "" ) ) (dict "commonName" (printf "%s-client" $fullname) "duration" (get (fromJson (include "_shims.time_Duration_String" (dict "a" (list (get (fromJson (include "_shims.time_ParseDuration" (dict "a" (list $duration) ))) "r")) ))) "r") "isCA" false "secretName" (printf "%s-client" $fullname) "privateKey" (mustMergeOverwrite (dict ) (dict "algorithm" "ECDSA" "size" (256 | int) )) "issuerRef" $issuerRef )) ))))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_chart.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_chart.go.tpl new file mode 100644 index 0000000000..46f6423e12 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_chart.go.tpl @@ -0,0 +1,62 @@ +{{- /* Generated from "chart.go" */ -}} + +{{- define "redpanda.render" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $manifests := (list (get (fromJson (include "redpanda.NodePortService" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.PodDisruptionBudget" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.ServiceAccount" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.ServiceInternal" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.ServiceMonitor" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.SidecarControllersRole" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.SidecarControllersRoleBinding" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.StatefulSet" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.PostInstallUpgradeJob" (dict "a" (list $dot) ))) "r")) -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.ConfigMaps" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.CertIssuers" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.RootCAs" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.ClientCerts" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.ClusterRoleBindings" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.ClusterRoles" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.LoadBalancerServices" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $obj := (get (fromJson (include "redpanda.Secrets" (dict "a" (list $dot) ))) "r") -}} +{{- $manifests = (concat (default (list ) $manifests) (list $obj)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $manifests = (concat (default (list ) $manifests) (default (list ) (get (fromJson (include "redpanda.consoleChartIntegration" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $manifests) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_configmap.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_configmap.go.tpl new file mode 100644 index 0000000000..ddcdcd8c7b --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_configmap.go.tpl @@ -0,0 +1,531 @@ +{{- /* Generated from "configmap.tpl.go" */ -}} + +{{- define "redpanda.ConfigMaps" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $cms := (list (get (fromJson (include "redpanda.RedpandaConfigMap" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.RPKProfile" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $cms) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaConfigMap" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "kind" "ConfigMap" "apiVersion" "v1" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "data" (dict "bootstrap.yaml" (get (fromJson (include "redpanda.BootstrapFile" (dict "a" (list $dot) ))) "r") "redpanda.yaml" (get (fromJson (include "redpanda.RedpandaConfigFile" (dict "a" (list $dot true) ))) "r") ) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapFile" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $bootstrap := (dict "kafka_enable_authorization" (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r") "enable_sasl" (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r") "enable_rack_awareness" $values.rackAwareness.enabled "storage_min_free_bytes" ((get (fromJson (include "redpanda.Storage.StorageMinFreeBytes" (dict "a" (list $values.storage) ))) "r") | int64) ) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.AuditLogging.Translate" (dict "a" (list $values.auditLogging $dot (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r")) ))) "r")) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.Logging.Translate" (dict "a" (list $values.logging) ))) "r")) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.TunableConfig.Translate" (dict "a" (list $values.config.tunable) ))) "r")) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.ClusterConfig.Translate" (dict "a" (list $values.config.cluster) ))) "r")) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.Auth.Translate" (dict "a" (list $values.auth (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r")) ))) "r")) -}} +{{- $bootstrap = (merge (dict ) $bootstrap (get (fromJson (include "redpanda.TieredStorageConfig.Translate" (dict "a" (list (deepCopy (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r")) $values.storage.tiered.credentialsSecretRef) ))) "r")) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $values.config.cluster "default_topic_replications" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_1 := $tmp_tuple_1.T2 -}} +{{- if (and (not $ok_1) (ge ($values.statefulset.replicas | int) (3 | int))) -}} +{{- $_ := (set $bootstrap "default_topic_replications" (3 | int)) -}} +{{- end -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $values.config.cluster "storage_min_free_bytes" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_2 := $tmp_tuple_2.T2 -}} +{{- if (not $ok_2) -}} +{{- $_ := (set $bootstrap "storage_min_free_bytes" ((get (fromJson (include "redpanda.Storage.StorageMinFreeBytes" (dict "a" (list $values.storage) ))) "r") | int64)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (toYaml $bootstrap)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaConfigFile" -}} +{{- $dot := (index .a 0) -}} +{{- $includeSeedServer := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $redpanda := (dict "empty_seed_starts_cluster" false ) -}} +{{- if $includeSeedServer -}} +{{- $_ := (set $redpanda "seed_servers" (get (fromJson (include "redpanda.Listeners.CreateSeedServers" (dict "a" (list $values.listeners ($values.statefulset.replicas | int) (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) ))) "r")) -}} +{{- end -}} +{{- $redpanda = (merge (dict ) $redpanda (get (fromJson (include "redpanda.NodeConfig.Translate" (dict "a" (list $values.config.node) ))) "r")) -}} +{{- $_ := (get (fromJson (include "redpanda.configureListeners" (dict "a" (list $redpanda $dot) ))) "r") -}} +{{- $redpandaYaml := (dict "redpanda" $redpanda "schema_registry" (get (fromJson (include "redpanda.schemaRegistry" (dict "a" (list $dot) ))) "r") "schema_registry_client" (get (fromJson (include "redpanda.kafkaClient" (dict "a" (list $dot) ))) "r") "pandaproxy" (get (fromJson (include "redpanda.pandaProxyListener" (dict "a" (list $dot) ))) "r") "pandaproxy_client" (get (fromJson (include "redpanda.kafkaClient" (dict "a" (list $dot) ))) "r") "rpk" (get (fromJson (include "redpanda.rpkNodeConfig" (dict "a" (list $dot) ))) "r") "config_file" "/etc/redpanda/redpanda.yaml" ) -}} +{{- if (and (and (get (fromJson (include "redpanda.RedpandaAtLeast_23_3_0" (dict "a" (list $dot) ))) "r") $values.auditLogging.enabled) (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r")) -}} +{{- $_ := (set $redpandaYaml "audit_log_client" (get (fromJson (include "redpanda.kafkaClient" (dict "a" (list $dot) ))) "r")) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (toYaml $redpandaYaml)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RPKProfile" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.external.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "kind" "ConfigMap" "apiVersion" "v1" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-rpk" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "data" (dict "profile" (toYaml (get (fromJson (include "redpanda.rpkProfile" (dict "a" (list $dot) ))) "r")) ) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkProfile" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $brokerList := (list ) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) (($values.statefulset.replicas | int)|int) (1|int) -}} +{{- $brokerList = (concat (default (list ) $brokerList) (list (printf "%s:%d" (get (fromJson (include "redpanda.advertisedHost" (dict "a" (list $dot $i) ))) "r") (((get (fromJson (include "redpanda.advertisedKafkaPort" (dict "a" (list $dot $i) ))) "r") | int) | int)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $adminAdvertisedList := (list ) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) (($values.statefulset.replicas | int)|int) (1|int) -}} +{{- $adminAdvertisedList = (concat (default (list ) $adminAdvertisedList) (list (printf "%s:%d" (get (fromJson (include "redpanda.advertisedHost" (dict "a" (list $dot $i) ))) "r") (((get (fromJson (include "redpanda.advertisedAdminPort" (dict "a" (list $dot $i) ))) "r") | int) | int)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $kafkaTLS := (get (fromJson (include "redpanda.rpkKafkaClientTLSConfiguration" (dict "a" (list $dot) ))) "r") -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $kafkaTLS "ca_file" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_3 := $tmp_tuple_3.T2 -}} +{{- if $ok_3 -}} +{{- $_ := (set $kafkaTLS "ca_file" "ca.crt") -}} +{{- end -}} +{{- $adminTLS := (get (fromJson (include "redpanda.rpkAdminAPIClientTLSConfiguration" (dict "a" (list $dot) ))) "r") -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $adminTLS "ca_file" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_4 := $tmp_tuple_4.T2 -}} +{{- if $ok_4 -}} +{{- $_ := (set $adminTLS "ca_file" "ca.crt") -}} +{{- end -}} +{{- $ka := (dict "brokers" $brokerList "tls" (coalesce nil) ) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $kafkaTLS) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $ka "tls" $kafkaTLS) -}} +{{- end -}} +{{- $aa := (dict "addresses" $adminAdvertisedList "tls" (coalesce nil) ) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $adminTLS) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $aa "tls" $adminTLS) -}} +{{- end -}} +{{- $result := (dict "name" (get (fromJson (include "redpanda.getFirstExternalKafkaListener" (dict "a" (list $dot) ))) "r") "kafka_api" $ka "admin_api" $aa ) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.advertisedKafkaPort" -}} +{{- $dot := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $externalKafkaListenerName := (get (fromJson (include "redpanda.getFirstExternalKafkaListener" (dict "a" (list $dot) ))) "r") -}} +{{- $listener := (index $values.listeners.kafka.external $externalKafkaListenerName) -}} +{{- $port := (($values.listeners.kafka.port | int) | int) -}} +{{- if (gt (($listener.port | int) | int) ((1 | int) | int)) -}} +{{- $port = (($listener.port | int) | int) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = ((index $listener.advertisedPorts $i) | int) -}} +{{- else -}}{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = ((index $listener.advertisedPorts (0 | int)) | int) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $port) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.advertisedAdminPort" -}} +{{- $dot := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $keys := (keys $values.listeners.admin.external) -}} +{{- $_ := (sortAlpha $keys) -}} +{{- $externalAdminListenerName := (first $keys) -}} +{{- $listener := (index $values.listeners.admin.external (get (fromJson (include "_shims.typeassertion" (dict "a" (list "string" $externalAdminListenerName) ))) "r")) -}} +{{- $port := (($values.listeners.admin.port | int) | int) -}} +{{- if (gt (($listener.port | int) | int) (1 | int)) -}} +{{- $port = (($listener.port | int) | int) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = ((index $listener.advertisedPorts $i) | int) -}} +{{- else -}}{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = ((index $listener.advertisedPorts (0 | int)) | int) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $port) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.advertisedHost" -}} +{{- $dot := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $address := (printf "%s-%d" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") ($i | int)) -}} +{{- if (ne (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.domain "") ))) "r") "") -}} +{{- $address = (printf "%s.%s" $address (tpl $values.external.domain $dot)) -}} +{{- end -}} +{{- if (le ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (0 | int)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $address) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (1 | int)) -}} +{{- $address = (index $values.external.addresses (0 | int)) -}} +{{- else -}} +{{- $address = (index $values.external.addresses $i) -}} +{{- end -}} +{{- if (ne (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.domain "") ))) "r") "") -}} +{{- $address = (printf "%s.%s" $address (tpl $values.external.domain $dot)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $address) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.getFirstExternalKafkaListener" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $keys := (keys $values.listeners.kafka.external) -}} +{{- $_ := (sortAlpha $keys) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "_shims.typeassertion" (dict "a" (list "string" (first $keys)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BrokerList" -}} +{{- $dot := (index .a 0) -}} +{{- $replicas := (index .a 1) -}} +{{- $port := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $bl := (coalesce nil) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) ($replicas|int) (1|int) -}} +{{- $bl = (concat (default (list ) $bl) (list (printf "%s-%d.%s:%d" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $i (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r") $port))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $bl) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkNodeConfig" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $brokerList := (get (fromJson (include "redpanda.BrokerList" (dict "a" (list $dot ($values.statefulset.replicas | int) ($values.listeners.kafka.port | int)) ))) "r") -}} +{{- $adminTLS := (coalesce nil) -}} +{{- $tls_5 := (get (fromJson (include "redpanda.rpkAdminAPIClientTLSConfiguration" (dict "a" (list $dot) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_5) ))) "r") | int) (0 | int)) -}} +{{- $adminTLS = $tls_5 -}} +{{- end -}} +{{- $brokerTLS := (coalesce nil) -}} +{{- $tls_6 := (get (fromJson (include "redpanda.rpkKafkaClientTLSConfiguration" (dict "a" (list $dot) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_6) ))) "r") | int) (0 | int)) -}} +{{- $brokerTLS = $tls_6 -}} +{{- end -}} +{{- $schemaRegistryTLS := (coalesce nil) -}} +{{- $tls_7 := (get (fromJson (include "redpanda.rpkSchemaRegistryClientTLSConfiguration" (dict "a" (list $dot) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_7) ))) "r") | int) (0 | int)) -}} +{{- $schemaRegistryTLS = $tls_7 -}} +{{- end -}} +{{- $result := (dict "overprovisioned" (get (fromJson (include "redpanda.RedpandaResources.GetOverProvisionValue" (dict "a" (list $values.resources) ))) "r") "enable_memory_locking" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.resources.memory.enable_memory_locking false) ))) "r") "additional_start_flags" (get (fromJson (include "redpanda.RedpandaAdditionalStartFlags" (dict "a" (list $dot ((get (fromJson (include "redpanda.RedpandaSMP" (dict "a" (list $dot) ))) "r") | int64)) ))) "r") "kafka_api" (dict "brokers" $brokerList "tls" $brokerTLS ) "admin_api" (dict "addresses" (get (fromJson (include "redpanda.Listeners.AdminList" (dict "a" (list $values.listeners ($values.statefulset.replicas | int) (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) ))) "r") "tls" $adminTLS ) "schema_registry" (dict "addresses" (get (fromJson (include "redpanda.Listeners.SchemaRegistryList" (dict "a" (list $values.listeners ($values.statefulset.replicas | int) (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) ))) "r") "tls" $schemaRegistryTLS ) ) -}} +{{- $result = (merge (dict ) $result (get (fromJson (include "redpanda.Tuning.Translate" (dict "a" (list $values.tuning) ))) "r")) -}} +{{- $result = (merge (dict ) $result (get (fromJson (include "redpanda.Config.CreateRPKConfiguration" (dict "a" (list $values.config) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkKafkaClientTLSConfiguration" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tls := $values.listeners.kafka.tls -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $tls $values.tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $result := (dict "ca_file" (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $tls $values.tls) ))) "r") ) -}} +{{- if $tls.requireClientAuth -}} +{{- $_ := (set $result "cert_file" (printf "%s/%s-client/tls.crt" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_ := (set $result "key_file" (printf "%s/%s-client/tls.key" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkAdminAPIClientTLSConfiguration" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tls := $values.listeners.admin.tls -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $tls $values.tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $result := (dict "ca_file" (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $tls $values.tls) ))) "r") ) -}} +{{- if $tls.requireClientAuth -}} +{{- $_ := (set $result "cert_file" (printf "%s/%s-client/tls.crt" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_ := (set $result "key_file" (printf "%s/%s-client/tls.key" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkSchemaRegistryClientTLSConfiguration" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tls := $values.listeners.schemaRegistry.tls -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $tls $values.tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $result := (dict "ca_file" (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $tls $values.tls) ))) "r") ) -}} +{{- if $tls.requireClientAuth -}} +{{- $_ := (set $result "cert_file" (printf "%s/%s-client/tls.crt" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_ := (set $result "key_file" (printf "%s/%s-client/tls.key" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.kafkaClient" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $brokerList := (list ) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) (($values.statefulset.replicas | int)|int) (1|int) -}} +{{- $brokerList = (concat (default (list ) $brokerList) (list (dict "address" (printf "%s-%d.%s" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $i (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) "port" ($values.listeners.kafka.port | int) ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $kafkaTLS := $values.listeners.kafka.tls -}} +{{- $brokerTLS := (coalesce nil) -}} +{{- if (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.kafka.tls $values.tls) ))) "r") -}} +{{- $brokerTLS = (dict "enabled" true "require_client_auth" $kafkaTLS.requireClientAuth "truststore_file" (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $kafkaTLS $values.tls) ))) "r") ) -}} +{{- if $kafkaTLS.requireClientAuth -}} +{{- $_ := (set $brokerTLS "cert_file" (printf "%s/%s-client/tls.crt" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- $_ := (set $brokerTLS "key_file" (printf "%s/%s-client/tls.key" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) -}} +{{- end -}} +{{- end -}} +{{- $cfg := (dict "brokers" $brokerList ) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $brokerTLS) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $cfg "broker_tls" $brokerTLS) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $cfg) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.configureListeners" -}} +{{- $redpanda := (index .a 0) -}} +{{- $dot := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_ := (set $redpanda "admin" (get (fromJson (include "redpanda.AdminListeners.Listeners" (dict "a" (list $values.listeners.admin) ))) "r")) -}} +{{- $_ := (set $redpanda "kafka_api" (get (fromJson (include "redpanda.KafkaListeners.Listeners" (dict "a" (list $values.listeners.kafka $values.auth) ))) "r")) -}} +{{- $_ := (set $redpanda "rpc_server" (get (fromJson (include "redpanda.rpcListeners" (dict "a" (list $dot) ))) "r")) -}} +{{- $_ := (set $redpanda "admin_api_tls" (coalesce nil)) -}} +{{- $tls_8 := (get (fromJson (include "redpanda.AdminListeners.ListenersTLS" (dict "a" (list $values.listeners.admin $values.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_8) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $redpanda "admin_api_tls" $tls_8) -}} +{{- end -}} +{{- $_ := (set $redpanda "kafka_api_tls" (coalesce nil)) -}} +{{- $tls_9 := (get (fromJson (include "redpanda.KafkaListeners.ListenersTLS" (dict "a" (list $values.listeners.kafka $values.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_9) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $redpanda "kafka_api_tls" $tls_9) -}} +{{- end -}} +{{- $tls_10 := (get (fromJson (include "redpanda.rpcListenersTLS" (dict "a" (list $dot) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_10) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $redpanda "rpc_server_tls" $tls_10) -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.pandaProxyListener" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $pandaProxy := (dict ) -}} +{{- $_ := (set $pandaProxy "pandaproxy_api" (get (fromJson (include "redpanda.HTTPListeners.Listeners" (dict "a" (list $values.listeners.http (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r")) ))) "r")) -}} +{{- $_ := (set $pandaProxy "pandaproxy_api_tls" (coalesce nil)) -}} +{{- $tls_11 := (get (fromJson (include "redpanda.HTTPListeners.ListenersTLS" (dict "a" (list $values.listeners.http $values.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_11) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $pandaProxy "pandaproxy_api_tls" $tls_11) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $pandaProxy) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.schemaRegistry" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $schemaReg := (dict ) -}} +{{- $_ := (set $schemaReg "schema_registry_api" (get (fromJson (include "redpanda.SchemaRegistryListeners.Listeners" (dict "a" (list $values.listeners.schemaRegistry (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r")) ))) "r")) -}} +{{- $_ := (set $schemaReg "schema_registry_api_tls" (coalesce nil)) -}} +{{- $tls_12 := (get (fromJson (include "redpanda.SchemaRegistryListeners.ListenersTLS" (dict "a" (list $values.listeners.schemaRegistry $values.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $tls_12) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $schemaReg "schema_registry_api_tls" $tls_12) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $schemaReg) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpcListenersTLS" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $r := $values.listeners.rpc -}} +{{- if (and (not ((or (or (get (fromJson (include "redpanda.RedpandaAtLeast_22_2_atleast_22_2_10" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.RedpandaAtLeast_22_3_atleast_22_3_13" (dict "a" (list $dot) ))) "r")) (get (fromJson (include "redpanda.RedpandaAtLeast_23_1_2" (dict "a" (list $dot) ))) "r")))) ((or (and (eq (toJson $r.tls.enabled) "null") $values.tls.enabled) (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $r.tls.enabled false) ))) "r")))) -}} +{{- $_ := (fail (printf "Redpanda version v%s does not support TLS on the RPC port. Please upgrade. See technical service bulletin 2023-01." (trimPrefix "v" (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")))) -}} +{{- end -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $r.tls $values.tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $certName := $r.tls.cert -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $certName) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $certName) "require_client_auth" $r.tls.requireClientAuth "truststore_file" (get (fromJson (include "redpanda.InternalTLS.TrustStoreFilePath" (dict "a" (list $r.tls $values.tls) ))) "r") )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpcListeners" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "address" "0.0.0.0" "port" ($values.listeners.rpc.port | int) )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.createInternalListenerTLSCfg" -}} +{{- $tls := (index .a 0) -}} +{{- $internal := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $internal $tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "name" "internal" "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $internal.cert) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $internal.cert) "require_client_auth" $internal.requireClientAuth "truststore_file" (get (fromJson (include "redpanda.InternalTLS.TrustStoreFilePath" (dict "a" (list $internal $tls) ))) "r") )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.createInternalListenerCfg" -}} +{{- $port := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "name" "internal" "address" "0.0.0.0" "port" $port )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAdditionalStartFlags" -}} +{{- $dot := (index .a 0) -}} +{{- $smp := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $chartFlags := (dict "smp" (printf "%d" ($smp | int)) "memory" (printf "%dM" (((get (fromJson (include "redpanda.RedpandaMemory" (dict "a" (list $dot) ))) "r") | int64) | int)) "reserve-memory" (printf "%dM" (((get (fromJson (include "redpanda.RedpandaReserveMemory" (dict "a" (list $dot) ))) "r") | int64) | int)) "default-log-level" $values.logging.logLevel ) -}} +{{- if (eq (index $values.config.node "developer_mode") true) -}} +{{- $_ := (unset $chartFlags "reserve-memory") -}} +{{- end -}} +{{- range $flag, $_ := $chartFlags -}} +{{- range $_, $userFlag := $values.statefulset.additionalRedpandaCmdFlags -}} +{{- if (regexMatch (printf "^--%s" $flag) $userFlag) -}} +{{- $_ := (unset $chartFlags $flag) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $keys := (keys $chartFlags) -}} +{{- $_ := (sortAlpha $keys) -}} +{{- $flags := (list ) -}} +{{- range $_, $key := $keys -}} +{{- $flags = (concat (default (list ) $flags) (list (printf "--%s=%s" $key (index $chartFlags $key)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $flags) (default (list ) $values.statefulset.additionalRedpandaCmdFlags))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_console.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_console.go.tpl new file mode 100644 index 0000000000..c158b4ef98 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_console.go.tpl @@ -0,0 +1,178 @@ +{{- /* Generated from "console.tpl.go" */ -}} + +{{- define "redpanda.consoleChartIntegration" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.console.enabled true) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $consoleDot := (index $dot.Subcharts "console") -}} +{{- $loadedValues := $consoleDot.Values -}} +{{- $consoleValue := $consoleDot.Values -}} +{{- $license_1 := (get (fromJson (include "redpanda.GetLicenseLiteral" (dict "a" (list $dot) ))) "r") -}} +{{- if (and (ne $license_1 "") (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.console.secret.create false) ))) "r"))) -}} +{{- $_ := (set $consoleValue.secret "create" true) -}} +{{- $_ := (set $consoleValue.secret "enterprise" (mustMergeOverwrite (dict ) (dict "license" $license_1 ))) -}} +{{- end -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.console.configmap.create false) ))) "r")) -}} +{{- $_ := (set $consoleValue.configmap "create" true) -}} +{{- $_ := (set $consoleValue.console "config" (get (fromJson (include "redpanda.ConsoleConfig" (dict "a" (list $dot) ))) "r")) -}} +{{- end -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.console.deployment.create false) ))) "r")) -}} +{{- $_ := (set $consoleValue.deployment "create" true) -}} +{{- if (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r") -}} +{{- $command := (list "sh" "-c" (printf "%s%s" (printf "%s%s" (printf "%s%s" (printf "%s%s" (printf "%s%s" (printf "%s%s" (printf "%s%s" "set -e; IFS=':' read -r KAFKA_SASL_USERNAME KAFKA_SASL_PASSWORD KAFKA_SASL_MECHANISM < <(grep \"\" $(find /mnt/users/* -print));" (printf " KAFKA_SASL_MECHANISM=${KAFKA_SASL_MECHANISM:-%s};" (get (fromJson (include "redpanda.SASLMechanism" (dict "a" (list $dot) ))) "r"))) " export KAFKA_SASL_USERNAME KAFKA_SASL_PASSWORD KAFKA_SASL_MECHANISM;") " export KAFKA_SCHEMAREGISTRY_USERNAME=$KAFKA_SASL_USERNAME;") " export KAFKA_SCHEMAREGISTRY_PASSWORD=$KAFKA_SASL_PASSWORD;") " export REDPANDA_ADMINAPI_USERNAME=$KAFKA_SASL_USERNAME;") " export REDPANDA_ADMINAPI_PASSWORD=$KAFKA_SASL_PASSWORD;") " /app/console $@") " --") -}} +{{- $_ := (set $consoleValue.deployment "command" $command) -}} +{{- end -}} +{{- $secret_2 := (get (fromJson (include "redpanda.GetLicenseSecretReference" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $secret_2) "null") -}} +{{- $_ := (set $consoleValue "enterprise" (mustMergeOverwrite (dict "licenseSecretRef" (dict "name" "" "key" "" ) ) (dict "licenseSecretRef" (mustMergeOverwrite (dict "name" "" "key" "" ) (dict "name" $secret_2.name "key" $secret_2.key )) ))) -}} +{{- end -}} +{{- $_ := (set $consoleValue "extraVolumes" (get (fromJson (include "redpanda.consoleTLSVolumes" (dict "a" (list $dot) ))) "r")) -}} +{{- $_ := (set $consoleValue "extraVolumeMounts" (get (fromJson (include "redpanda.consoleTLSVolumesMounts" (dict "a" (list $dot) ))) "r")) -}} +{{- $_ := (set $consoleDot "Values" $consoleValue) -}} +{{- $cfg := (get (fromJson (include "console.ConfigMap" (dict "a" (list $consoleDot) ))) "r") -}} +{{- if (eq (toJson $consoleValue.podAnnotations) "null") -}} +{{- $_ := (set $consoleValue "podAnnotations" (dict )) -}} +{{- end -}} +{{- $_ := (set $consoleValue.podAnnotations "checksum-redpanda-chart/config" (sha256sum (toYaml $cfg))) -}} +{{- end -}} +{{- $_ := (set $consoleDot "Values" $consoleValue) -}} +{{- $manifests := (list (get (fromJson (include "console.Secret" (dict "a" (list $consoleDot) ))) "r") (get (fromJson (include "console.ConfigMap" (dict "a" (list $consoleDot) ))) "r") (get (fromJson (include "console.Deployment" (dict "a" (list $consoleDot) ))) "r")) -}} +{{- $_ := (set $consoleDot "Values" $loadedValues) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $manifests) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.consoleTLSVolumesMounts" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $mounts := (list ) -}} +{{- $sasl_3 := $values.auth.sasl -}} +{{- if (and $sasl_3.enabled (ne $sasl_3.secretRef "")) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf "%s-users" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "mountPath" "/mnt/users" "readOnly" true )))) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list (get (fromJson (include "redpanda.Listeners.TrustStores" (dict "a" (list $values.listeners $values.tls) ))) "r")) ))) "r") | int) (0 | int)) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "truststores" "mountPath" "/etc/truststores" "readOnly" true )))) -}} +{{- end -}} +{{- $visitedCert := (dict ) -}} +{{- range $_, $tlsCfg := (list $values.listeners.kafka.tls $values.listeners.schemaRegistry.tls $values.listeners.admin.tls) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $visitedCert $tlsCfg.cert (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $visited := $tmp_tuple_1.T2 -}} +{{- if (or (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $tlsCfg $values.tls) ))) "r")) $visited) -}} +{{- continue -}} +{{- end -}} +{{- $_ := (set $visitedCert $tlsCfg.cert true) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf "redpanda-%s-cert" $tlsCfg.cert) "mountPath" (printf "%s/%s" "/etc/tls/certs" $tlsCfg.cert) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $mounts) (default (list ) $values.console.extraVolumeMounts))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.consoleTLSVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $volumes := (list ) -}} +{{- $sasl_4 := $values.auth.sasl -}} +{{- if (and $sasl_4.enabled (ne $sasl_4.secretRef "")) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" $values.auth.sasl.secretRef )) )) (dict "name" (printf "%s-users" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) )))) -}} +{{- end -}} +{{- $vol_5 := (get (fromJson (include "redpanda.Listeners.TrustStoreVolume" (dict "a" (list $values.listeners $values.tls) ))) "r") -}} +{{- if (ne (toJson $vol_5) "null") -}} +{{- $volumes = (concat (default (list ) $volumes) (list $vol_5)) -}} +{{- end -}} +{{- $visitedCert := (dict ) -}} +{{- range $_, $tlsCfg := (list $values.listeners.kafka.tls $values.listeners.schemaRegistry.tls $values.listeners.admin.tls) -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $visitedCert $tlsCfg.cert (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $visited := $tmp_tuple_2.T2 -}} +{{- if (or (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $tlsCfg $values.tls) ))) "r")) $visited) -}} +{{- continue -}} +{{- end -}} +{{- $_ := (set $visitedCert $tlsCfg.cert true) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "defaultMode" (0o420 | int) "secretName" (get (fromJson (include "redpanda.CertSecretName" (dict "a" (list $dot $tlsCfg.cert (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $values.tls.certs) $tlsCfg.cert) ))) "r")) ))) "r") )) )) (dict "name" (printf "redpanda-%s-cert" $tlsCfg.cert) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $volumes) (default (list ) $values.console.extraVolumes))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ConsoleConfig" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $schemaURLs := (coalesce nil) -}} +{{- if $values.listeners.schemaRegistry.enabled -}} +{{- $schema := "http" -}} +{{- if (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.schemaRegistry.tls $values.tls) ))) "r") -}} +{{- $schema = "https" -}} +{{- end -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) (($values.statefulset.replicas | int)|int) (1|int) -}} +{{- $schemaURLs = (concat (default (list ) $schemaURLs) (list (printf "%s://%s-%d.%s:%d" $schema (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $i (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r") ($values.listeners.schemaRegistry.port | int)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $schema := "http" -}} +{{- if (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.admin.tls $values.tls) ))) "r") -}} +{{- $schema = "https" -}} +{{- end -}} +{{- $c := (dict "kafka" (dict "brokers" (get (fromJson (include "redpanda.BrokerList" (dict "a" (list $dot ($values.statefulset.replicas | int) ($values.listeners.kafka.port | int)) ))) "r") "sasl" (dict "enabled" (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r") ) "tls" (get (fromJson (include "redpanda.KafkaListeners.ConsoleTLS" (dict "a" (list $values.listeners.kafka $values.tls) ))) "r") "schemaRegistry" (dict "enabled" $values.listeners.schemaRegistry.enabled "urls" $schemaURLs "tls" (get (fromJson (include "redpanda.SchemaRegistryListeners.ConsoleTLS" (dict "a" (list $values.listeners.schemaRegistry $values.tls) ))) "r") ) ) "redpanda" (dict "adminApi" (dict "enabled" true "urls" (list (printf "%s://%s:%d" $schema (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r") ($values.listeners.admin.port | int))) "tls" (get (fromJson (include "redpanda.AdminListeners.ConsoleTLS" (dict "a" (list $values.listeners.admin $values.tls) ))) "r") ) ) ) -}} +{{- if $values.connectors.enabled -}} +{{- $port := (dig "connectors" "connectors" "restPort" (8083 | int) $dot.Values.AsMap) -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.asintegral" (dict "a" (list $port) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_3.T2 -}} +{{- $p := ($tmp_tuple_3.T1 | int) -}} +{{- if (not $ok) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $c) | toJson -}} +{{- break -}} +{{- end -}} +{{- $connectorsURL := (printf "http://%s.%s.svc.%s:%d" (get (fromJson (include "redpanda.ConnectorsFullName" (dict "a" (list $dot) ))) "r") $dot.Release.Namespace (trimSuffix "." $values.clusterDomain) $p) -}} +{{- $_ := (set $c "connect" (mustMergeOverwrite (dict "enabled" false "clusters" (coalesce nil) "connectTimeout" 0 "readTimeout" 0 "requestTimeout" 0 ) (dict "enabled" $values.connectors.enabled "clusters" (list (mustMergeOverwrite (dict "name" "" "url" "" "tls" (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false ) "username" "" "password" "" "token" "" ) (dict "name" "connectors" "url" $connectorsURL "tls" (mustMergeOverwrite (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false ) (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false )) "username" "" "password" "" "token" "" ))) ))) -}} +{{- end -}} +{{- if (eq (toJson $values.console.console) "null") -}} +{{- $_ := (set $values.console "console" (mustMergeOverwrite (dict ) (dict "config" (dict ) ))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $values.console.console.config $c)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ConnectorsFullName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (ne (dig "connectors" "connectors" "fullnameOverwrite" "" $dot.Values.AsMap) "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $values.connectors.connectors.fullnameOverwrite) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list (printf "%s-connectors" $dot.Release.Name)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_example-commands.tpl b/charts/redpanda/redpanda/5.9.10/templates/_example-commands.tpl new file mode 100644 index 0000000000..9a5c695e32 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_example-commands.tpl @@ -0,0 +1,58 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} + + +{{/* +Any rpk command that's given to the user in NOTES.txt must be defined in this template file +and tested in a test. +*/}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-acl-user-create" -}} +{{- $cmd := (get ((include "redpanda.RpkACLUserCreate" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-acl-create" -}} +{{- $cmd := (get ((include "redpanda.RpkACLCreate" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-cluster-info" -}} +{{- $cmd := (get ((include "redpanda.RpkClusterInfo" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-topic-create" -}} +{{- $cmd := (get ((include "redpanda.RpkTopicCreate" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-topic-describe" -}} +{{- $cmd := (get ((include "redpanda.RpkTopicDescribe" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} + +{{/* tested in tests/test-kafka-sasl-status.yaml */}} +{{- define "rpk-topic-delete" -}} +{{- $cmd := (get ((include "redpanda.RpkTopicDelete" (dict "a" (list .))) | fromJson) "r") }} +{{- $cmd }} +{{- end -}} \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/templates/_helpers.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_helpers.go.tpl new file mode 100644 index 0000000000..f7501789f8 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_helpers.go.tpl @@ -0,0 +1,570 @@ +{{- /* Generated from "helpers.go" */ -}} + +{{- define "redpanda.ChartLabel" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list (replace "+" "_" (printf "%s-%s" $dot.Chart.Name $dot.Chart.Version))) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Name" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "string" (index $dot.Values "nameOverride") "") ))) "r")) ))) "r") -}} +{{- $ok_2 := $tmp_tuple_1.T2 -}} +{{- $override_1 := $tmp_tuple_1.T1 -}} +{{- if (and $ok_2 (ne $override_1 "")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $override_1) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $dot.Chart.Name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Fullname" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "string" (index $dot.Values "fullnameOverride") "") ))) "r")) ))) "r") -}} +{{- $ok_4 := $tmp_tuple_2.T2 -}} +{{- $override_3 := $tmp_tuple_2.T1 -}} +{{- if (and $ok_4 (ne $override_3 "")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $override_3) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $dot.Release.Name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.FullLabels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $labels := (dict ) -}} +{{- if (ne (toJson $values.commonLabels) "null") -}} +{{- $labels = $values.commonLabels -}} +{{- end -}} +{{- $defaults := (dict "helm.sh/chart" (get (fromJson (include "redpanda.ChartLabel" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name "app.kubernetes.io/managed-by" $dot.Release.Service "app.kubernetes.io/component" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") ) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $labels $defaults)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ServiceAccountName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $serviceAccount := $values.serviceAccount -}} +{{- if (and $serviceAccount.create (ne $serviceAccount.name "")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $serviceAccount.name) | toJson -}} +{{- break -}} +{{- else -}}{{- if $serviceAccount.create -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) | toJson -}} +{{- break -}} +{{- else -}}{{- if (ne $serviceAccount.name "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $serviceAccount.name) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "default") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Tag" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tag := (toString $values.image.tag) -}} +{{- if (eq $tag "") -}} +{{- $tag = $dot.Chart.AppVersion -}} +{{- end -}} +{{- $pattern := "^v(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" -}} +{{- if (not (regexMatch $pattern $tag)) -}} +{{- $_ := (fail "image.tag must start with a 'v' and be a valid semver") -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tag) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ServiceName" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (and (ne (toJson $values.service) "null") (ne (toJson $values.service.name) "null")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.cleanForK8s" (dict "a" (list $values.service.name) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.InternalDomain" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $service := (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r") -}} +{{- $ns := $dot.Release.Namespace -}} +{{- $domain := (trimSuffix "." $values.clusterDomain) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s.%s.svc.%s." $service $ns $domain)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TLSEnabled" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if $values.tls.enabled -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $listeners := (list "kafka" "admin" "schemaRegistry" "rpc" "http") -}} +{{- range $_, $listener := $listeners -}} +{{- $tlsCert := (dig "listeners" $listener "tls" "cert" false $dot.Values.AsMap) -}} +{{- $tlsEnabled := (dig "listeners" $listener "tls" "enabled" false $dot.Values.AsMap) -}} +{{- if (and (not (empty $tlsEnabled)) (not (empty $tlsCert))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $external := (dig "listeners" $listener "external" false $dot.Values.AsMap) -}} +{{- if (empty $external) -}} +{{- continue -}} +{{- end -}} +{{- $keys := (keys (get (fromJson (include "_shims.typeassertion" (dict "a" (list (printf "map[%s]%s" "string" "interface {}") $external) ))) "r")) -}} +{{- range $_, $key := $keys -}} +{{- $enabled := (dig "listeners" $listener "external" $key "enabled" false $dot.Values.AsMap) -}} +{{- $tlsCert := (dig "listeners" $listener "external" $key "tls" "cert" false $dot.Values.AsMap) -}} +{{- $tlsEnabled := (dig "listeners" $listener "external" $key "tls" "enabled" false $dot.Values.AsMap) -}} +{{- if (and (and (not (empty $enabled)) (not (empty $tlsCert))) (not (empty $tlsEnabled))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" false) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ClientAuthRequired" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $listeners := (list "kafka" "admin" "schemaRegistry" "rpc" "http") -}} +{{- range $_, $listener := $listeners -}} +{{- $required := (dig "listeners" $listener "tls" "requireClientAuth" false $dot.Values.AsMap) -}} +{{- if (not (empty $required)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" false) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.DefaultMounts" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/etc/redpanda" )))) (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.CommonMounts" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $mounts := (list ) -}} +{{- $sasl_5 := $values.auth.sasl -}} +{{- if (and $sasl_5.enabled (ne $sasl_5.secretRef "")) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "users" "mountPath" "/etc/secrets/users" "readOnly" true )))) -}} +{{- end -}} +{{- if (get (fromJson (include "redpanda.TLSEnabled" (dict "a" (list $dot) ))) "r") -}} +{{- $certNames := (keys $values.tls.certs) -}} +{{- $_ := (sortAlpha $certNames) -}} +{{- range $_, $name := $certNames -}} +{{- $cert := (index $values.tls.certs $name) -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $cert.enabled true) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf "redpanda-%s-cert" $name) "mountPath" (printf "%s/%s" "/etc/tls/certs" $name) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $adminTLS := $values.listeners.admin.tls -}} +{{- if $adminTLS.requireClientAuth -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "mtls-client" "mountPath" (printf "%s/%s-client" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) )))) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $mounts) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.DefaultVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") )) (dict )) )) (dict "name" "base-config" )))) (default (list ) (get (fromJson (include "redpanda.CommonVolumes" (dict "a" (list $dot) ))) "r")))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.CommonVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $volumes := (list ) -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (get (fromJson (include "redpanda.TLSEnabled" (dict "a" (list $dot) ))) "r") -}} +{{- $certNames := (keys $values.tls.certs) -}} +{{- $_ := (sortAlpha $certNames) -}} +{{- range $_, $name := $certNames -}} +{{- $cert := (index $values.tls.certs $name) -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $cert.enabled true) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (get (fromJson (include "redpanda.CertSecretName" (dict "a" (list $dot $name $cert) ))) "r") "defaultMode" (0o440 | int) )) )) (dict "name" (printf "redpanda-%s-cert" $name) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $adminTLS := $values.listeners.admin.tls -}} +{{- $cert := (index $values.tls.certs $adminTLS.cert) -}} +{{- if $adminTLS.requireClientAuth -}} +{{- $secretName := (printf "%s-client" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- if (ne (toJson $cert.clientSecretRef) "null") -}} +{{- $secretName = $cert.clientSecretRef.name -}} +{{- end -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" $secretName "defaultMode" (0o440 | int) )) )) (dict "name" "mtls-client" )))) -}} +{{- end -}} +{{- end -}} +{{- $sasl_6 := $values.auth.sasl -}} +{{- if (and $sasl_6.enabled (ne $sasl_6.secretRef "")) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" $sasl_6.secretRef )) )) (dict "name" "users" )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $volumes) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.CertSecretName" -}} +{{- $dot := (index .a 0) -}} +{{- $certName := (index .a 1) -}} +{{- $cert := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $cert.secretRef) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $cert.secretRef.name) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s-%s-cert" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $certName)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.PodSecurityContext" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $sc := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.statefulset.podSecurityContext $values.statefulset.securityContext) ))) "r") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict ) (dict "fsGroup" $sc.fsGroup "fsGroupChangePolicy" $sc.fsGroupChangePolicy ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ContainerSecurityContext" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $sc := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.statefulset.podSecurityContext $values.statefulset.securityContext) ))) "r") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict ) (dict "runAsUser" $sc.runAsUser "runAsGroup" (get (fromJson (include "redpanda.coalesce" (dict "a" (list (list $sc.runAsGroup $sc.fsGroup)) ))) "r") "allowPrivilegeEscalation" (get (fromJson (include "redpanda.coalesce" (dict "a" (list (list $sc.allowPrivilegeEscalation $sc.allowPriviledgeEscalation)) ))) "r") "runAsNonRoot" $sc.runAsNonRoot ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_22_2_0" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=22.2.0-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_22_3_0" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=22.3.0-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_23_1_1" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=23.1.1-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_23_1_2" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=23.1.2-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_22_3_atleast_22_3_13" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=22.3.13-0,<22.4") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_22_2_atleast_22_2_10" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=22.2.10-0,<22.3") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_23_2_1" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=23.2.1-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaAtLeast_23_3_0" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.redpandaAtLeast" (dict "a" (list $dot ">=23.3.0-0 || <0.0.1-0") ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.redpandaAtLeast" -}} +{{- $dot := (index .a 0) -}} +{{- $constraint := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $version := (trimPrefix "v" (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (list (semverCompare $constraint $version) nil)) ))) "r") -}} +{{- $err := $tmp_tuple_3.T2 -}} +{{- $result := $tmp_tuple_3.T1 -}} +{{- if (ne (toJson $err) "null") -}} +{{- $_ := (fail $err) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.cleanForK8s" -}} +{{- $in := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (trimSuffix "-" (trunc (63 | int) $in))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaSMP" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $coresInMillies := ((get (fromJson (include "_shims.resource_MilliValue" (dict "a" (list $values.resources.cpu.cores) ))) "r") | int64) -}} +{{- if (lt $coresInMillies (1000 | int64)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (1 | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $values.resources.cpu.cores) ))) "r") | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.coalesce" -}} +{{- $values := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- range $_, $v := $values -}} +{{- if (ne (toJson $v) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $v) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StrategicMergePatch" -}} +{{- $overrides := (index .a 0) -}} +{{- $original := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $overrides.labels) "null") -}} +{{- $_ := (set $original.metadata "labels" (merge (dict ) $overrides.labels (default (dict ) $original.metadata.labels))) -}} +{{- end -}} +{{- if (ne (toJson $overrides.annotations) "null") -}} +{{- $_ := (set $original.metadata "annotations" (merge (dict ) $overrides.annotations (default (dict ) $original.metadata.annotations))) -}} +{{- end -}} +{{- if (ne (toJson $overrides.spec.securityContext) "null") -}} +{{- $_ := (set $original.spec "securityContext" (merge (dict ) $overrides.spec.securityContext (default (mustMergeOverwrite (dict ) (dict )) $original.spec.securityContext))) -}} +{{- end -}} +{{- if (ne (toJson $overrides.spec.automountServiceAccountToken) "null") -}} +{{- $_ := (set $original.spec "automountServiceAccountToken" $overrides.spec.automountServiceAccountToken) -}} +{{- end -}} +{{- $overrideContainers := (dict ) -}} +{{- range $i, $_ := $overrides.spec.containers -}} +{{- $container := (index $overrides.spec.containers $i) -}} +{{- $_ := (set $overrideContainers (toString $container.name) $container) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (and (ne (toJson $overrides.spec.volumes) "null") (gt ((get (fromJson (include "_shims.len" (dict "a" (list $overrides.spec.volumes) ))) "r") | int) (0 | int))) -}} +{{- $newVolumes := (list ) -}} +{{- $overrideVolumes := (dict ) -}} +{{- range $i, $_ := $overrides.spec.volumes -}} +{{- $vol := (index $overrides.spec.volumes $i) -}} +{{- $_ := (set $overrideVolumes $vol.name $vol) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $vol := $original.spec.volumes -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $overrideVolumes $vol.name (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_8 := $tmp_tuple_4.T2 -}} +{{- $overrideVol_7 := $tmp_tuple_4.T1 -}} +{{- if $ok_8 -}} +{{- $newVolumes = (concat (default (list ) $newVolumes) (list $overrideVol_7)) -}} +{{- $_ := (unset $overrideVolumes $vol.name) -}} +{{- continue -}} +{{- end -}} +{{- $newVolumes = (concat (default (list ) $newVolumes) (list $vol)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $vol := $overrideVolumes -}} +{{- $newVolumes = (concat (default (list ) $newVolumes) (list $vol)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $original.spec "volumes" $newVolumes) -}} +{{- end -}} +{{- $merged := (coalesce nil) -}} +{{- range $_, $container := $original.spec.containers -}} +{{- $tmp_tuple_5 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $overrideContainers $container.name (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_10 := $tmp_tuple_5.T2 -}} +{{- $override_9 := $tmp_tuple_5.T1 -}} +{{- if $ok_10 -}} +{{- $env := (concat (default (list ) $container.env) (default (list ) $override_9.env)) -}} +{{- $container = (merge (dict ) $override_9 $container) -}} +{{- $_ := (set $container "env" $env) -}} +{{- end -}} +{{- if (eq (toJson $container.env) "null") -}} +{{- $_ := (set $container "env" (list )) -}} +{{- end -}} +{{- $merged = (concat (default (list ) $merged) (list $container)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $original.spec "containers" $merged) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $original) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_helpers.tpl b/charts/redpanda/redpanda/5.9.10/templates/_helpers.tpl new file mode 100644 index 0000000000..a885f9dcd3 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_helpers.tpl @@ -0,0 +1,368 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{/* +Expand the name of the chart. +*/}} +{{- define "redpanda.name" -}} +{{- get ((include "redpanda.Name" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "redpanda.fullname" -}} +{{- get ((include "redpanda.Fullname" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* +Create a default service name +*/}} +{{- define "redpanda.servicename" -}} +{{- get ((include "redpanda.ServiceName" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* +full helm labels + common labels +*/}} +{{- define "full.labels" -}} +{{- (get ((include "redpanda.FullLabels" (dict "a" (list .))) | fromJson) "r") | toYaml }} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "redpanda.chart" -}} +{{- get ((include "redpanda.Chart" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "redpanda.serviceAccountName" -}} +{{- get ((include "redpanda.ServiceAccountName" (dict "a" (list .))) | fromJson) "r" }} +{{- end }} + +{{/* +Use AppVersion if image.tag is not set +*/}} +{{- define "redpanda.tag" -}} +{{- get ((include "redpanda.Tag" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* Generate internal fqdn */}} +{{- define "redpanda.internal.domain" -}} +{{- get ((include "redpanda.InternalDomain" (dict "a" (list .))) | fromJson) "r" }} +{{- end -}} + +{{/* ConfigMap variables */}} +{{- define "admin-internal-tls-enabled" -}} +{{- toJson (dict "bool" (get ((include "redpanda.InternalTLS.IsEnabled" (dict "a" (list .Values.listeners.admin.tls .Values.tls))) | fromJson) "r")) -}} +{{- end -}} + +{{- define "kafka-internal-tls-enabled" -}} +{{- $listener := .Values.listeners.kafka -}} +{{- toJson (dict "bool" (and (dig "tls" "enabled" .Values.tls.enabled $listener) (not (empty (dig "tls" "cert" "" $listener))))) -}} +{{- end -}} + +{{- define "kafka-external-tls-cert" -}} +{{- dig "tls" "cert" .Values.listeners.kafka.tls.cert .listener -}} +{{- end -}} + +{{- define "http-internal-tls-enabled" -}} +{{- $listener := .Values.listeners.http -}} +{{- toJson (dict "bool" (and (dig "tls" "enabled" .Values.tls.enabled $listener) (not (empty (dig "tls" "cert" "" $listener))))) -}} +{{- end -}} + +{{- define "schemaRegistry-internal-tls-enabled" -}} +{{- $listener := .Values.listeners.schemaRegistry -}} +{{- toJson (dict "bool" (and (dig "tls" "enabled" .Values.tls.enabled $listener) (not (empty (dig "tls" "cert" "" $listener))))) -}} +{{- end -}} + +{{- define "tls-enabled" -}} +{{- $tlsenabled := get ((include "redpanda.TLSEnabled" (dict "a" (list .))) | fromJson) "r" }} +{{- toJson (dict "bool" $tlsenabled) -}} +{{- end -}} + +{{- define "sasl-enabled" -}} +{{- toJson (dict "bool" (dig "enabled" false .Values.auth.sasl)) -}} +{{- end -}} + +{{- define "admin-api-urls" -}} +{{ printf "${SERVICE_NAME}.%s" (include "redpanda.internal.domain" .) }}:{{.Values.listeners.admin.port }} +{{- end -}} + +{{- define "admin-api-service-url" -}} +{{ include "redpanda.internal.domain" .}}:{{.Values.listeners.admin.port }} +{{- end -}} + +{{- define "sasl-mechanism" -}} +{{- dig "sasl" "mechanism" "SCRAM-SHA-512" .Values.auth -}} +{{- end -}} + +{{- define "fail-on-insecure-sasl-logging" -}} +{{- if (include "sasl-enabled" .|fromJson).bool -}} + {{- $check := list + (include "redpanda-atleast-23-1-1" .|fromJson).bool + (include "redpanda-22-3-atleast-22-3-13" .|fromJson).bool + (include "redpanda-22-2-atleast-22-2-10" .|fromJson).bool + -}} + {{- if not (mustHas true $check) -}} + {{- fail "SASL is enabled and the redpanda version specified leaks secrets to the logs. Please choose a newer version of redpanda." -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "fail-on-unsupported-helm-version" -}} + {{- $helmVer := (fromYaml (toYaml .Capabilities.HelmVersion)).version -}} + {{- if semverCompare "<3.8.0-0" $helmVer -}} + {{- fail (printf "helm version %s is not supported. Please use helm version v3.8.0 or newer." $helmVer) -}} + {{- end -}} +{{- end -}} + +{{- define "redpanda-atleast-22-2-0" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_22_2_0" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-atleast-22-3-0" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_22_3_0" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-atleast-23-1-1" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_23_1_1" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-atleast-23-1-2" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_23_1_2" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-22-3-atleast-22-3-13" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_22_3_atleast_22_3_13" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-22-2-atleast-22-2-10" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_22_2_atleast_22_2_10" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-atleast-23-2-1" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_23_2_1" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} +{{- define "redpanda-atleast-23-3-0" -}} +{{- toJson (dict "bool" (get ((include "redpanda.RedpandaAtLeast_23_3_0" (dict "a" (list .))) | fromJson) "r")) }} +{{- end -}} + +{{- define "redpanda-22-2-x-without-sasl" -}} +{{- $result := (include "redpanda-atleast-22-3-0" . | fromJson).bool -}} +{{- if or (include "sasl-enabled" . | fromJson).bool .Values.listeners.kafka.authenticationMethod -}} +{{- $result := false -}} +{{- end -}} +{{- toJson (dict "bool" $result) -}} +{{- end -}} + +{{- define "pod-security-context" -}} +{{- get ((include "redpanda.PodSecurityContext" (dict "a" (list .))) | fromJson) "r" | toYaml }} +{{- end -}} + +{{- define "container-security-context" -}} +{{- get ((include "redpanda.ContainerSecurityContext" (dict "a" (list .))) | fromJson) "r" | toYaml }} +{{- end -}} + +{{- define "admin-tls-curl-flags" -}} + {{- $result := "" -}} + {{- if (include "admin-internal-tls-enabled" . | fromJson).bool -}} + {{- $path := (printf "/etc/tls/certs/%s" .Values.listeners.admin.tls.cert) -}} + {{- $result = (printf "--cacert %s/tls.crt" $path) -}} + {{- if .Values.listeners.admin.tls.requireClientAuth -}} + {{- $result = (printf "--cacert %s/ca.crt --cert %s/tls.crt --key %s/tls.key" $path $path $path) -}} + {{- end -}} + {{- end -}} + {{- $result -}} +{{- end -}} + +{{- define "admin-http-protocol" -}} + {{- $result := "http" -}} + {{- if (include "admin-internal-tls-enabled" . | fromJson).bool -}} + {{- $result = "https" -}} + {{- end -}} + {{- $result -}} +{{- end -}} + +{{- /* +advertised-port returns either the only advertised port if only one is specified, +or the port specified for this pod ordinal when there is a full list provided. + +This will return a string int or panic if there is more than one port provided, +but not enough ports for the number of replicas requested. +*/ -}} +{{- define "advertised-port" -}} + {{- $port := dig "port" .listenerVals.port .externalVals -}} + {{- if .externalVals.advertisedPorts -}} + {{- if eq (len .externalVals.advertisedPorts) 1 -}} + {{- $port = mustFirst .externalVals.advertisedPorts -}} + {{- else -}} + {{- $port = index .externalVals.advertisedPorts .replicaIndex -}} + {{- end -}} + {{- end -}} + {{ $port }} +{{- end -}} + +{{- /* +advertised-host returns a json string with the data needed for configuring the advertised listener +*/ -}} +{{- define "advertised-host" -}} + {{- $host := dict "name" .externalName "address" .externalAdvertiseAddress "port" .port -}} + {{- if .values.external.addresses -}} + {{- $address := "" -}} + {{- if gt (len .values.external.addresses) 1 -}} + {{- $address = (index .values.external.addresses .replicaIndex) -}} + {{- else -}} + {{- $address = (index .values.external.addresses 0) -}} + {{- end -}} + {{- if ( .values.external.domain | default "" ) }} + {{- $host = dict "name" .externalName "address" (printf "%s.%s" $address .values.external.domain) "port" .port -}} + {{- else -}} + {{- $host = dict "name" .externalName "address" $address "port" .port -}} + {{- end -}} + {{- end -}} + {{- toJson $host -}} +{{- end -}} + +{{- define "is-licensed" -}} +{{- toJson (dict "bool" (or (not (empty (include "enterprise-license" . ))) (not (empty (include "enterprise-secret" . ))))) -}} +{{- end -}} + +{{- define "seed-server-list" -}} + {{- $brokers := list -}} + {{- range $ordinal := until (.Values.statefulset.replicas | int) -}} + {{- $brokers = append $brokers (printf "%s-%d.%s" + (include "redpanda.fullname" $) + $ordinal + (include "redpanda.internal.domain" $)) + -}} + {{- end -}} + {{- toJson $brokers -}} +{{- end -}} + +{{/* +return license checks deprecated values if current values is empty +*/}} +{{- define "enterprise-license" -}} +{{- if dig "license" dict .Values.enterprise -}} + {{- .Values.enterprise.license -}} +{{- else -}} + {{- .Values.license_key -}} +{{- end -}} +{{- end -}} + +{{/* +return licenseSecretRef checks deprecated values entry if current values empty +*/}} +{{- define "enterprise-secret" -}} +{{- if ( dig "licenseSecretRef" dict .Values.enterprise ) -}} + {{- .Values.enterprise.licenseSecretRef -}} +{{- else if not (empty .Values.license_secret_ref ) -}} + {{- .Values.license_secret_ref -}} +{{- end -}} +{{- end -}} + +{{/* +return licenseSecretRef.name checks deprecated values entry if current values empty +*/}} +{{- define "enterprise-secret-name" -}} +{{- if ( dig "licenseSecretRef" dict .Values.enterprise ) -}} + {{- dig "name" "" .Values.enterprise.licenseSecretRef -}} +{{- else if not (empty .Values.license_secret_ref ) -}} + {{- dig "secret_name" "" .Values.license_secret_ref -}} +{{- end -}} +{{- end -}} + +{{/* +return licenseSecretRef.key checks deprecated values entry if current values empty +*/}} +{{- define "enterprise-secret-key" -}} +{{- if ( dig "licenseSecretRef" dict .Values.enterprise ) -}} + {{- dig "key" "" .Values.enterprise.licenseSecretRef -}} +{{- else if not (empty .Values.license_secret_ref ) -}} + {{- dig "secret_key" "" .Values.license_secret_ref -}} +{{- end -}} +{{- end -}} + +{{/* mounts that are common to all containers */}} +{{- define "common-mounts" -}} +{{- $mounts := get ((include "redpanda.CommonMounts" (dict "a" (list .))) | fromJson) "r" }} +{{- if $mounts -}} +{{- toYaml $mounts -}} +{{- end -}} +{{- end -}} + +{{/* mounts that are common to most containers */}} +{{- define "default-mounts" -}} +{{- $mounts := get ((include "redpanda.DefaultMounts" (dict "a" (list .))) | fromJson) "r" }} +{{- if $mounts -}} +{{- toYaml $mounts -}} +{{- end -}} +{{- end -}} + +{{/* volumes that are common to all pods */}} +{{- define "common-volumes" -}} +{{- $volumes := get ((include "redpanda.CommonVolumes" (dict "a" (list .))) | fromJson) "r" }} +{{- if $volumes -}} +{{- toYaml $volumes -}} +{{- end -}} +{{- end -}} + +{{/* the default set of volumes for most pods, except the sts pod */}} +{{- define "default-volumes" -}} +{{- $volumes := get ((include "redpanda.DefaultVolumes" (dict "a" (list .))) | fromJson) "r" }} +{{- if $volumes -}} +{{- toYaml $volumes -}} +{{- end -}} +{{- end -}} + +{{/* support legacy storage.tieredConfig */}} +{{- define "storage-tiered-config" -}} +{{- $cfg := get ((include "redpanda.StorageTieredConfig" (dict "a" (list .))) | fromJson) "r" }} +{{- if $cfg -}} +{{- toYaml $cfg -}} +{{- end -}} +{{- end -}} + +{{/* + rpk sasl environment variables + + this will return a string with the correct environment variables to use for SASL based on the + version of the redpada container being used +*/}} +{{- define "rpk-sasl-environment-variables" -}} +{{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool -}} +RPK_USER RPK_PASS RPK_SASL_MECHANISM +{{- else -}} +REDPANDA_SASL_USERNAME REDPANDA_SASL_PASSWORD REDPANDA_SASL_MECHANISM +{{- end -}} +{{- end -}} + +{{- define "curl-options" -}} +{{- print " -svm3 --fail --retry \"120\" --retry-max-time \"120\" --retry-all-errors -o - -w \"\\nstatus=%{http_code} %{redirect_url} size=%{size_download} time=%{time_total} content-type=\\\"%{content_type}\\\"\\n\" "}} +{{- end -}} + +{{- define "advertised-address-template" -}} + {{- $prefixTemplate := dig "prefixTemplate" "" .externalListener -}} + {{- if empty $prefixTemplate -}} + {{- $prefixTemplate = dig "prefixTemplate" "" .externalVals -}} + {{- end -}} + {{ quote $prefixTemplate }} +{{- end -}} + +{{/* check if client auth is enabled for any of the listeners */}} +{{- define "client-auth-required" -}} +{{- $requireClientAuth := get ((include "redpanda.ClientAuthRequired" (dict "a" (list .))) | fromJson) "r" }} +{{- toJson (dict "bool" $requireClientAuth) -}} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/_memory.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_memory.go.tpl new file mode 100644 index 0000000000..015a771b46 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_memory.go.tpl @@ -0,0 +1,63 @@ +{{- /* Generated from "memory.go" */ -}} + +{{- define "redpanda.RedpandaReserveMemory" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $rpMem_1 := $values.resources.memory.redpanda -}} +{{- if (and (ne (toJson $rpMem_1) "null") (ne (toJson $rpMem_1.reserveMemory) "null")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" ((div ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $rpMem_1.reserveMemory) ))) "r") | int64) ((mul (1024 | int) (1024 | int)))) | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" ((add (((mulf (((get (fromJson (include "redpanda.ContainerMemory" (dict "a" (list $dot) ))) "r") | int64) | float64) 0.002) | float64) | int64) (200 | int64)) | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaMemory" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $memory := ((0 | int64) | int64) -}} +{{- $containerMemory := ((get (fromJson (include "redpanda.ContainerMemory" (dict "a" (list $dot) ))) "r") | int64) -}} +{{- $rpMem_2 := $values.resources.memory.redpanda -}} +{{- if (and (ne (toJson $rpMem_2) "null") (ne (toJson $rpMem_2.memory) "null")) -}} +{{- $memory = ((div ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $rpMem_2.memory) ))) "r") | int64) ((mul (1024 | int) (1024 | int)))) | int64) -}} +{{- else -}} +{{- $memory = (((mulf ($containerMemory | float64) 0.8) | float64) | int64) -}} +{{- end -}} +{{- if (eq $memory (0 | int64)) -}} +{{- $_ := (fail "unable to get memory value redpanda-memory") -}} +{{- end -}} +{{- if (lt $memory (256 | int64)) -}} +{{- $_ := (fail (printf "%d is below the minimum value for Redpanda" $memory)) -}} +{{- end -}} +{{- if (gt ((add $memory ((get (fromJson (include "redpanda.RedpandaReserveMemory" (dict "a" (list $dot) ))) "r") | int64)) | int64) $containerMemory) -}} +{{- $_ := (fail (printf "Not enough container memory for Redpanda memory values where Redpanda: %d, reserve: %d, container: %d" $memory ((get (fromJson (include "redpanda.RedpandaReserveMemory" (dict "a" (list $dot) ))) "r") | int64) $containerMemory)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $memory) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ContainerMemory" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (ne (toJson $values.resources.memory.container.min) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" ((div ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $values.resources.memory.container.min) ))) "r") | int64) ((mul (1024 | int) (1024 | int)))) | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" ((div ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $values.resources.memory.container.max) ))) "r") | int64) ((mul (1024 | int) (1024 | int)))) | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_notes.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_notes.go.tpl new file mode 100644 index 0000000000..e547ce092d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_notes.go.tpl @@ -0,0 +1,167 @@ +{{- /* Generated from "notes.go" */ -}} + +{{- define "redpanda.Warnings" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $warnings := (coalesce nil) -}} +{{- $w_1 := (get (fromJson (include "redpanda.cpuWarning" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne $w_1 "") -}} +{{- $warnings = (concat (default (list ) $warnings) (list (printf `**Warning**: %s` $w_1))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $warnings) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.cpuWarning" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $coresInMillis := ((get (fromJson (include "_shims.resource_MilliValue" (dict "a" (list $values.resources.cpu.cores) ))) "r") | int64) -}} +{{- if (lt $coresInMillis (1000 | int64)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%dm is below the minimum recommended CPU value for Redpanda" $coresInMillis)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Notes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $anySASL := (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $values.auth) ))) "r") -}} +{{- $notes := (coalesce nil) -}} +{{- $notes = (concat (default (list ) $notes) (list `` `` `` `` (printf `Congratulations on installing %s!` $dot.Chart.Name) `` `The pods will rollout in a few seconds. To check the status:` `` (printf ` kubectl -n %s rollout status statefulset %s --watch` $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")))) -}} +{{- if (and $values.external.enabled (eq $values.external.type "LoadBalancer")) -}} +{{- $notes = (concat (default (list ) $notes) (list `` `If you are using the load balancer service with a cloud provider, the services will likely have automatically-generated addresses. In this scenario the advertised listeners must be updated in order for external access to work. Run the following command once Redpanda is deployed:` `` (printf ` helm upgrade %s redpanda/redpanda --reuse-values -n %s --set $(kubectl get svc -n %s -o jsonpath='{"external.addresses={"}{ range .items[*]}{.status.loadBalancer.ingress[0].ip }{.status.loadBalancer.ingress[0].hostname}{","}{ end }{"}\n"}')` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") $dot.Release.Namespace $dot.Release.Namespace))) -}} +{{- end -}} +{{- $profiles := (keys $values.listeners.kafka.external) -}} +{{- $_ := (sortAlpha $profiles) -}} +{{- $profileName := (index $profiles (0 | int)) -}} +{{- $notes = (concat (default (list ) $notes) (list `` `Set up rpk for access to your external listeners:`)) -}} +{{- $profile := (index $values.listeners.kafka.external $profileName) -}} +{{- if (get (fromJson (include "redpanda.TLSEnabled" (dict "a" (list $dot) ))) "r") -}} +{{- $external := "" -}} +{{- if (and (ne (toJson $profile.tls) "null") (ne (toJson $profile.tls.cert) "null")) -}} +{{- $external = $profile.tls.cert -}} +{{- else -}} +{{- $external = $values.listeners.kafka.tls.cert -}} +{{- end -}} +{{- $notes = (concat (default (list ) $notes) (list (printf ` kubectl get secret -n %s %s-%s-cert -o go-template='{{ index .data "ca.crt" | base64decode }}' > ca.crt` $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $external))) -}} +{{- if (or $values.listeners.kafka.tls.requireClientAuth $values.listeners.admin.tls.requireClientAuth) -}} +{{- $notes = (concat (default (list ) $notes) (list (printf ` kubectl get secret -n %s %s-client -o go-template='{{ index .data "tls.crt" | base64decode }}' > tls.crt` $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) (printf ` kubectl get secret -n %s %s-client -o go-template='{{ index .data "tls.key" | base64decode }}' > tls.key` $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")))) -}} +{{- end -}} +{{- end -}} +{{- $notes = (concat (default (list ) $notes) (list (printf ` rpk profile create --from-profile <(kubectl get configmap -n %s %s-rpk -o go-template='{{ .data.profile }}') %s` $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $profileName) `` `Set up dns to look up the pods on their Kubernetes Nodes. You can use this query to get the list of short-names to IP addresses. Add your external domain to the hostnames and you could test by adding these to your /etc/hosts:` `` (printf ` kubectl get pod -n %s -o custom-columns=node:.status.hostIP,name:.metadata.name --no-headers -l app.kubernetes.io/name=redpanda,app.kubernetes.io/component=redpanda-statefulset` $dot.Release.Namespace))) -}} +{{- if $anySASL -}} +{{- $notes = (concat (default (list ) $notes) (list `` `Set the credentials in the environment:` `` (printf ` kubectl -n %s get secret %s -o go-template="{{ range .data }}{{ . | base64decode }}{{ end }}" | IFS=: read -r %s` $dot.Release.Namespace $values.auth.sasl.secretRef (get (fromJson (include "redpanda.RpkSASLEnvironmentVariables" (dict "a" (list $dot) ))) "r")) (printf ` export %s` (get (fromJson (include "redpanda.RpkSASLEnvironmentVariables" (dict "a" (list $dot) ))) "r")))) -}} +{{- end -}} +{{- $notes = (concat (default (list ) $notes) (list `` `Try some sample commands:`)) -}} +{{- if $anySASL -}} +{{- $notes = (concat (default (list ) $notes) (list `Create a user:` `` (printf ` %s` (get (fromJson (include "redpanda.RpkACLUserCreate" (dict "a" (list $dot) ))) "r")) `` `Give the user permissions:` `` (printf ` %s` (get (fromJson (include "redpanda.RpkACLCreate" (dict "a" (list $dot) ))) "r")))) -}} +{{- end -}} +{{- $notes = (concat (default (list ) $notes) (list `` `Get the api status:` `` (printf ` %s` (get (fromJson (include "redpanda.RpkClusterInfo" (dict "a" (list $dot) ))) "r")) `` `Create a topic` `` (printf ` %s` (get (fromJson (include "redpanda.RpkTopicCreate" (dict "a" (list $dot) ))) "r")) `` `Describe the topic:` `` (printf ` %s` (get (fromJson (include "redpanda.RpkTopicDescribe" (dict "a" (list $dot) ))) "r")) `` `Delete the topic:` `` (printf ` %s` (get (fromJson (include "redpanda.RpkTopicDelete" (dict "a" (list $dot) ))) "r")))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $notes) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkACLUserCreate" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf `rpk acl user create myuser --new-password changeme --mechanism %s` (get (fromJson (include "redpanda.SASLMechanism" (dict "a" (list $dot) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SASLMechanism" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (ne (toJson $values.auth.sasl) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $values.auth.sasl.mechanism) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "SCRAM-SHA-512") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkACLCreate" -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" `rpk acl create --allow-principal 'myuser' --allow-host '*' --operation all --topic 'test-topic'`) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkClusterInfo" -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" `rpk cluster info`) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkTopicCreate" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf `rpk topic create test-topic -p 3 -r %d` (min (3 | int64) (($values.statefulset.replicas | int) | int64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkTopicDescribe" -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" `rpk topic describe test-topic`) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkTopicDelete" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" `rpk topic delete test-topic`) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RpkSASLEnvironmentVariables" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (get (fromJson (include "redpanda.RedpandaAtLeast_23_2_1" (dict "a" (list $dot) ))) "r") -}} +{{- $_is_returning = true -}} +{{- (dict "r" `RPK_USER RPK_PASS RPK_SASL_MECHANISM`) | toJson -}} +{{- break -}} +{{- else -}} +{{- $_is_returning = true -}} +{{- (dict "r" `REDPANDA_SASL_USERNAME REDPANDA_SASL_PASSWORD REDPANDA_SASL_MECHANISM`) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_poddisruptionbudget.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_poddisruptionbudget.go.tpl new file mode 100644 index 0000000000..763b7b0bdf --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_poddisruptionbudget.go.tpl @@ -0,0 +1,21 @@ +{{- /* Generated from "poddisruptionbudget.go" */ -}} + +{{- define "redpanda.PodDisruptionBudget" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $budget := ($values.statefulset.budget.maxUnavailable | int) -}} +{{- $minReplicas := ((div ($values.statefulset.replicas | int) (2 | int)) | int) -}} +{{- if (and (gt $budget (1 | int)) (gt $budget $minReplicas)) -}} +{{- $_ := (fail (printf "statefulset.budget.maxUnavailable is set too high to maintain quorum: %d > %d" $budget $minReplicas)) -}} +{{- end -}} +{{- $maxUnavailable := ($budget | int) -}} +{{- $matchLabels := (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") -}} +{{- $_ := (set $matchLabels "redpanda.com/poddisruptionbudget" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "disruptionsAllowed" 0 "currentHealthy" 0 "desiredHealthy" 0 "expectedPods" 0 ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "policy/v1" "kind" "PodDisruptionBudget" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict ) (dict "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" $matchLabels )) "maxUnavailable" $maxUnavailable )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_post-install-upgrade-job.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_post-install-upgrade-job.go.tpl new file mode 100644 index 0000000000..617d6dbf30 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_post-install-upgrade-job.go.tpl @@ -0,0 +1,123 @@ +{{- /* Generated from "post_install_upgrade_job.go" */ -}} + +{{- define "redpanda.bootstrapYamlTemplater" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $env := (get (fromJson (include "redpanda.TieredStorageCredentials.AsEnvVars" (dict "a" (list $values.storage.tiered.credentialsSecretRef (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r")) ))) "r") -}} +{{- $image := (printf `%s:%s` $values.statefulset.sideCars.controllers.image.repository $values.statefulset.sideCars.controllers.image.tag) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "bootstrap-yaml-envsubst" "image" $image "command" (list "/redpanda-operator" "envsubst" "/tmp/base-config/bootstrap.yaml" "--output" "/tmp/config/.bootstrap.yaml") "env" $env "resources" (mustMergeOverwrite (dict ) (dict "limits" (dict "cpu" (get (fromJson (include "_shims.resource_MustParse" (dict "a" (list "100m") ))) "r") "memory" (get (fromJson (include "_shims.resource_MustParse" (dict "a" (list "125Mi") ))) "r") ) "requests" (dict "cpu" (get (fromJson (include "_shims.resource_MustParse" (dict "a" (list "100m") ))) "r") "memory" (get (fromJson (include "_shims.resource_MustParse" (dict "a" (list "125Mi") ))) "r") ) )) "securityContext" (mustMergeOverwrite (dict ) (dict "allowPrivilegeEscalation" false "readOnlyRootFilesystem" true "runAsNonRoot" true )) "volumeMounts" (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "config" "mountPath" "/tmp/config/" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/tmp/base-config/" ))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.PostInstallUpgradeJob" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.post_install_job.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $image := (printf `%s:%s` $values.statefulset.sideCars.controllers.image.repository $values.statefulset.sideCars.controllers.image.tag) -}} +{{- $job := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "batch/v1" "kind" "Job" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-configuration" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (merge (dict ) (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") (default (dict ) $values.post_install_job.labels)) "annotations" (merge (dict ) (dict "helm.sh/hook" "post-install,post-upgrade" "helm.sh/hook-delete-policy" "before-hook-creation" "helm.sh/hook-weight" "-5" ) (default (dict ) $values.post_install_job.annotations)) )) "spec" (mustMergeOverwrite (dict "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) ) (dict "template" (get (fromJson (include "redpanda.StrategicMergePatch" (dict "a" (list $values.post_install_job.podTemplate (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "generateName" (printf "%s-post-" $dot.Release.Name) "labels" (merge (dict ) (dict "app.kubernetes.io/name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name "app.kubernetes.io/component" (printf "%.50s-post-install" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r")) ) (default (dict ) $values.commonLabels)) )) "spec" (mustMergeOverwrite (dict "containers" (coalesce nil) ) (dict "nodeSelector" $values.nodeSelector "affinity" (get (fromJson (include "redpanda.postInstallJobAffinity" (dict "a" (list $dot) ))) "r") "tolerations" (get (fromJson (include "redpanda.tolerations" (dict "a" (list $dot) ))) "r") "restartPolicy" "Never" "securityContext" (get (fromJson (include "redpanda.PodSecurityContext" (dict "a" (list $dot) ))) "r") "imagePullSecrets" (default (coalesce nil) $values.imagePullSecrets) "initContainers" (list (get (fromJson (include "redpanda.bootstrapYamlTemplater" (dict "a" (list $dot) ))) "r")) "automountServiceAccountToken" false "containers" (list (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "post-install" "image" $image "env" (get (fromJson (include "redpanda.PostInstallUpgradeEnvironmentVariables" (dict "a" (list $dot) ))) "r") "command" (list "/redpanda-operator" "sync-cluster-config" "--users-directory" "/etc/secrets/users" "--redpanda-yaml" "/tmp/base-config/redpanda.yaml" "--bootstrap-yaml" "/tmp/config/.bootstrap.yaml") "resources" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.post_install_job.resources (mustMergeOverwrite (dict ) (dict ))) ))) "r") "securityContext" (merge (dict ) (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.post_install_job.securityContext (mustMergeOverwrite (dict ) (dict ))) ))) "r") (get (fromJson (include "redpanda.ContainerSecurityContext" (dict "a" (list $dot) ))) "r")) "volumeMounts" (concat (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "config" "mountPath" "/tmp/config" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/tmp/base-config" )))) ))) "volumes" (concat (default (list ) (get (fromJson (include "redpanda.CommonVolumes" (dict "a" (list $dot) ))) "r")) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") )) (dict )) )) (dict "name" "base-config" )) (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "emptyDir" (mustMergeOverwrite (dict ) (dict )) )) (dict "name" "config" )))) "serviceAccountName" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") )) ))) ))) "r") )) )) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $job) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.postInstallJobAffinity" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (empty $values.post_install_job.affinity)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $values.post_install_job.affinity) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $values.post_install_job.affinity $values.affinity)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.tolerations" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $result := (coalesce nil) -}} +{{- range $_, $t := $values.tolerations -}} +{{- $result = (concat (default (list ) $result) (list (merge (dict ) $t))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.PostInstallUpgradeEnvironmentVariables" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $envars := (list ) -}} +{{- $license_1 := (get (fromJson (include "redpanda.GetLicenseLiteral" (dict "a" (list $dot) ))) "r") -}} +{{- $secretReference_2 := (get (fromJson (include "redpanda.GetLicenseSecretReference" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne $license_1 "") -}} +{{- $envars = (concat (default (list ) $envars) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_LICENSE" "value" $license_1 )))) -}} +{{- else -}}{{- if (ne (toJson $secretReference_2) "null") -}} +{{- $envars = (concat (default (list ) $envars) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_LICENSE" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" $secretReference_2 )) )))) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.bootstrapEnvVars" (dict "a" (list $dot $envars) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.GetLicenseLiteral" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (ne $values.enterprise.license "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $values.enterprise.license) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $values.license_key) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.GetLicenseSecretReference" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (empty $values.enterprise.licenseSecretRef)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" $values.enterprise.licenseSecretRef.name )) (dict "key" $values.enterprise.licenseSecretRef.key ))) | toJson -}} +{{- break -}} +{{- else -}}{{- if (not (empty $values.license_secret_ref)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" $values.license_secret_ref.secret_name )) (dict "key" $values.license_secret_ref.secret_key ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_post_upgrade_job.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_post_upgrade_job.go.tpl new file mode 100644 index 0000000000..6a95bb94e6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_post_upgrade_job.go.tpl @@ -0,0 +1,87 @@ +{{- /* Generated from "post_upgrade_job.go" */ -}} + +{{- define "redpanda.PostUpgrade" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.post_upgrade_job.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $labels := (default (dict ) $values.post_upgrade_job.labels) -}} +{{- $annotations := (default (dict ) $values.post_upgrade_job.annotations) -}} +{{- $annotations = (merge (dict ) (dict "helm.sh/hook" "post-upgrade" "helm.sh/hook-delete-policy" "before-hook-creation" "helm.sh/hook-weight" "-10" ) $annotations) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) ) "status" (dict ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "batch/v1" "kind" "Job" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-post-upgrade" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (merge (dict ) (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") $labels) "annotations" $annotations )) "spec" (mustMergeOverwrite (dict "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) ) (dict "backoffLimit" $values.post_upgrade_job.backoffLimit "template" (get (fromJson (include "redpanda.StrategicMergePatch" (dict "a" (list $values.post_upgrade_job.podTemplate (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $dot.Release.Name "labels" (merge (dict ) (dict "app.kubernetes.io/name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name "app.kubernetes.io/component" (printf "%s-post-upgrade" (trunc (50 | int) (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r"))) ) $values.commonLabels) )) "spec" (mustMergeOverwrite (dict "containers" (coalesce nil) ) (dict "nodeSelector" $values.nodeSelector "affinity" (merge (dict ) $values.post_upgrade_job.affinity $values.affinity) "tolerations" $values.tolerations "restartPolicy" "Never" "securityContext" (get (fromJson (include "redpanda.PodSecurityContext" (dict "a" (list $dot) ))) "r") "serviceAccountName" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "imagePullSecrets" (default (coalesce nil) $values.imagePullSecrets) "containers" (list (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "post-upgrade" "image" (printf "%s:%s" $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "command" (list "/bin/bash" "-c") "args" (list (get (fromJson (include "redpanda.PostUpgradeJobScript" (dict "a" (list $dot) ))) "r")) "env" (get (fromJson (include "redpanda.rpkEnvVars" (dict "a" (list $dot $values.post_upgrade_job.extraEnv) ))) "r") "envFrom" $values.post_upgrade_job.extraEnvFrom "securityContext" (merge (dict ) (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.post_upgrade_job.securityContext (mustMergeOverwrite (dict ) (dict ))) ))) "r") (get (fromJson (include "redpanda.ContainerSecurityContext" (dict "a" (list $dot) ))) "r")) "resources" $values.post_upgrade_job.resources "volumeMounts" (get (fromJson (include "redpanda.DefaultMounts" (dict "a" (list $dot) ))) "r") ))) "volumes" (get (fromJson (include "redpanda.DefaultVolumes" (dict "a" (list $dot) ))) "r") )) ))) ))) "r") )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.PostUpgradeJobScript" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $script := (list `set -e` ``) -}} +{{- range $key, $value := $values.config.cluster -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.asintegral" (dict "a" (list $value) ))) "r")) ))) "r") -}} +{{- $isInt64 := $tmp_tuple_1.T2 -}} +{{- $asInt64 := ($tmp_tuple_1.T1 | int64) -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "bool" $value false) ))) "r")) ))) "r") -}} +{{- $ok_2 := $tmp_tuple_2.T2 -}} +{{- $asBool_1 := $tmp_tuple_2.T1 -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "string" $value "") ))) "r")) ))) "r") -}} +{{- $ok_4 := $tmp_tuple_3.T2 -}} +{{- $asStr_3 := $tmp_tuple_3.T1 -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list (printf "[]%s" "interface {}") $value (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_6 := $tmp_tuple_4.T2 -}} +{{- $asSlice_5 := $tmp_tuple_4.T1 -}} +{{- if (and $ok_2 $asBool_1) -}} +{{- $script = (concat (default (list ) $script) (list (printf "rpk cluster config set %s %t" $key $asBool_1))) -}} +{{- else -}}{{- if (and $ok_4 (ne $asStr_3 "")) -}} +{{- $script = (concat (default (list ) $script) (list (printf "rpk cluster config set %s %s" $key $asStr_3))) -}} +{{- else -}}{{- if (and $isInt64 (gt $asInt64 (0 | int64))) -}} +{{- $script = (concat (default (list ) $script) (list (printf "rpk cluster config set %s %d" $key $asInt64))) -}} +{{- else -}}{{- if (and $ok_6 (gt ((get (fromJson (include "_shims.len" (dict "a" (list $asSlice_5) ))) "r") | int) (0 | int))) -}} +{{- $script = (concat (default (list ) $script) (list (printf `rpk cluster config set %s "[ %s ]"` $key (join "," $asSlice_5)))) -}} +{{- else -}}{{- if (not (empty $value)) -}} +{{- $script = (concat (default (list ) $script) (list (printf "rpk cluster config set %s %v" $key $value))) -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $tmp_tuple_5 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $values.config.cluster "default_topic_replications" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_7 := $tmp_tuple_5.T2 -}} +{{- if (and (not $ok_7) (ge ($values.statefulset.replicas | int) (3 | int))) -}} +{{- $script = (concat (default (list ) $script) (list "rpk cluster config set default_topic_replications 3")) -}} +{{- end -}} +{{- $tmp_tuple_6 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $values.config.cluster "storage_min_free_bytes" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_8 := $tmp_tuple_6.T2 -}} +{{- if (not $ok_8) -}} +{{- $script = (concat (default (list ) $script) (list (printf "rpk cluster config set storage_min_free_bytes %d" ((get (fromJson (include "redpanda.Storage.StorageMinFreeBytes" (dict "a" (list $values.storage) ))) "r") | int64)))) -}} +{{- end -}} +{{- if (get (fromJson (include "redpanda.RedpandaAtLeast_23_2_1" (dict "a" (list $dot) ))) "r") -}} +{{- $service := $values.listeners.admin -}} +{{- $caCert := "" -}} +{{- $scheme := "http" -}} +{{- if (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $service.tls $values.tls) ))) "r") -}} +{{- $scheme = "https" -}} +{{- $caCert = (printf "--cacert %q" (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $service.tls $values.tls) ))) "r")) -}} +{{- end -}} +{{- $url := (printf "%s://%s:%d/v1/debug/restart_service?service=schema-registry" $scheme (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r") (($service.port | int) | int64)) -}} +{{- $script = (concat (default (list ) $script) (list `if [ -d "/etc/secrets/users/" ]; then` ` IFS=":" read -r USER_NAME PASSWORD MECHANISM < <(grep "" $(find /etc/secrets/users/* -print))` ` curl -svm3 --fail --retry "120" --retry-max-time "120" --retry-all-errors --ssl-reqd \` (printf ` %s \` $caCert) ` -X PUT -u ${USER_NAME}:${PASSWORD} \` (printf ` %s || true` $url) `fi`)) -}} +{{- end -}} +{{- $script = (concat (default (list ) $script) (list "")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (join "\n" $script)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_rbac.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_rbac.go.tpl new file mode 100644 index 0000000000..162092626d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_rbac.go.tpl @@ -0,0 +1,116 @@ +{{- /* Generated from "rbac.go" */ -}} + +{{- define "redpanda.ClusterRoles" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $crs := (coalesce nil) -}} +{{- $cr_1 := (get (fromJson (include "redpanda.SidecarControllersClusterRole" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $cr_1) "null") -}} +{{- $crs = (concat (default (list ) $crs) (list $cr_1)) -}} +{{- end -}} +{{- if (not $values.rbac.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $crs) | toJson -}} +{{- break -}} +{{- end -}} +{{- $rpkBundleName := (printf "%s-rpk-bundle" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $crs = (concat (default (list ) $crs) (default (list ) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "rules" (coalesce nil) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRole" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "rules" (list (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "nodes") "verbs" (list "get" "list") ))) )) (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "rules" (coalesce nil) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRole" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $rpkBundleName "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "rules" (list (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "configmaps" "endpoints" "events" "limitranges" "persistentvolumeclaims" "pods" "pods/log" "replicationcontrollers" "resourcequotas" "serviceaccounts" "services") "verbs" (list "get" "list") ))) ))))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $crs) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ClusterRoleBindings" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $crbs := (coalesce nil) -}} +{{- $crb_2 := (get (fromJson (include "redpanda.SidecarControllersClusterRoleBinding" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $crb_2) "null") -}} +{{- $crbs = (concat (default (list ) $crbs) (list $crb_2)) -}} +{{- end -}} +{{- if (not $values.rbac.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $crbs) | toJson -}} +{{- break -}} +{{- end -}} +{{- $rpkBundleName := (printf "%s-rpk-bundle" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $crbs = (concat (default (list ) $crbs) (default (list ) (list (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "roleRef" (dict "apiGroup" "" "kind" "" "name" "" ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRoleBinding" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "roleRef" (mustMergeOverwrite (dict "apiGroup" "" "kind" "" "name" "" ) (dict "apiGroup" "rbac.authorization.k8s.io" "kind" "ClusterRole" "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") )) "subjects" (list (mustMergeOverwrite (dict "kind" "" "name" "" ) (dict "kind" "ServiceAccount" "name" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace ))) )) (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "roleRef" (dict "apiGroup" "" "kind" "" "name" "" ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRoleBinding" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $rpkBundleName "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "roleRef" (mustMergeOverwrite (dict "apiGroup" "" "kind" "" "name" "" ) (dict "apiGroup" "rbac.authorization.k8s.io" "kind" "ClusterRole" "name" $rpkBundleName )) "subjects" (list (mustMergeOverwrite (dict "kind" "" "name" "" ) (dict "kind" "ServiceAccount" "name" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace ))) ))))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $crbs) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SidecarControllersClusterRole" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.statefulset.sideCars.controllers.enabled) (not $values.statefulset.sideCars.controllers.createRBAC)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $sidecarControllerName := (printf "%s-sidecar-controllers" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "rules" (coalesce nil) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRole" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $sidecarControllerName "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "rules" (list (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "nodes") "verbs" (list "get" "list" "watch") )) (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "persistentvolumes") "verbs" (list "delete" "get" "list" "patch" "update" "watch") ))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SidecarControllersClusterRoleBinding" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.statefulset.sideCars.controllers.enabled) (not $values.statefulset.sideCars.controllers.createRBAC)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $sidecarControllerName := (printf "%s-sidecar-controllers" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "roleRef" (dict "apiGroup" "" "kind" "" "name" "" ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "ClusterRoleBinding" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $sidecarControllerName "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "roleRef" (mustMergeOverwrite (dict "apiGroup" "" "kind" "" "name" "" ) (dict "apiGroup" "rbac.authorization.k8s.io" "kind" "ClusterRole" "name" $sidecarControllerName )) "subjects" (list (mustMergeOverwrite (dict "kind" "" "name" "" ) (dict "kind" "ServiceAccount" "name" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace ))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SidecarControllersRole" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.statefulset.sideCars.controllers.enabled) (not $values.statefulset.sideCars.controllers.createRBAC)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $sidecarControllerName := (printf "%s-sidecar-controllers" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "rules" (coalesce nil) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "Role" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $sidecarControllerName "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "rules" (list (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "apps") "resources" (list "statefulsets/status") "verbs" (list "patch" "update") )) (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "secrets" "pods") "verbs" (list "get" "list" "watch") )) (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "apps") "resources" (list "statefulsets") "verbs" (list "get" "patch" "update" "list" "watch") )) (mustMergeOverwrite (dict "verbs" (coalesce nil) ) (dict "apiGroups" (list "") "resources" (list "persistentvolumeclaims") "verbs" (list "delete" "get" "list" "patch" "update" "watch") ))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SidecarControllersRoleBinding" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.statefulset.sideCars.controllers.enabled) (not $values.statefulset.sideCars.controllers.createRBAC)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $sidecarControllerName := (printf "%s-sidecar-controllers" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "roleRef" (dict "apiGroup" "" "kind" "" "name" "" ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "rbac.authorization.k8s.io/v1" "kind" "RoleBinding" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $sidecarControllerName "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "roleRef" (mustMergeOverwrite (dict "apiGroup" "" "kind" "" "name" "" ) (dict "apiGroup" "rbac.authorization.k8s.io" "kind" "Role" "name" $sidecarControllerName )) "subjects" (list (mustMergeOverwrite (dict "kind" "" "name" "" ) (dict "kind" "ServiceAccount" "name" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace ))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_secrets.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_secrets.go.tpl new file mode 100644 index 0000000000..5e95f3f82c --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_secrets.go.tpl @@ -0,0 +1,419 @@ +{{- /* Generated from "secrets.go" */ -}} + +{{- define "redpanda.Secrets" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $secrets := (coalesce nil) -}} +{{- $secrets = (concat (default (list ) $secrets) (list (get (fromJson (include "redpanda.SecretSTSLifecycle" (dict "a" (list $dot) ))) "r"))) -}} +{{- $saslUsers_1 := (get (fromJson (include "redpanda.SecretSASLUsers" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $saslUsers_1) "null") -}} +{{- $secrets = (concat (default (list ) $secrets) (list $saslUsers_1)) -}} +{{- end -}} +{{- $configWatcher_2 := (get (fromJson (include "redpanda.SecretConfigWatcher" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $configWatcher_2) "null") -}} +{{- $secrets = (concat (default (list ) $secrets) (list $configWatcher_2)) -}} +{{- end -}} +{{- $secrets = (concat (default (list ) $secrets) (list (get (fromJson (include "redpanda.SecretConfigurator" (dict "a" (list $dot) ))) "r"))) -}} +{{- $fsValidator_3 := (get (fromJson (include "redpanda.SecretFSValidator" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $fsValidator_3) "null") -}} +{{- $secrets = (concat (default (list ) $secrets) (list $fsValidator_3)) -}} +{{- end -}} +{{- $bootstrapUser_4 := (get (fromJson (include "redpanda.SecretBootstrapUser" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $bootstrapUser_4) "null") -}} +{{- $secrets = (concat (default (list ) $secrets) (list $bootstrapUser_4)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secrets) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretSTSLifecycle" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $secret := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-sts-lifecycle" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict ) )) -}} +{{- $adminCurlFlags := (get (fromJson (include "redpanda.adminTLSCurlFlags" (dict "a" (list $dot) ))) "r") -}} +{{- $_ := (set $secret.stringData "common.sh" (join "\n" (list `#!/usr/bin/env bash` `` `# the SERVICE_NAME comes from the metadata.name of the pod, essentially the POD_NAME` (printf `CURL_URL="%s"` (get (fromJson (include "redpanda.adminInternalURL" (dict "a" (list $dot) ))) "r")) `` `# commands used throughout` (printf `CURL_NODE_ID_CMD="curl --silent --fail %s ${CURL_URL}/v1/node_config"` $adminCurlFlags) `` `CURL_MAINTENANCE_DELETE_CMD_PREFIX='curl -X DELETE --silent -o /dev/null -w "%{http_code}"'` `CURL_MAINTENANCE_PUT_CMD_PREFIX='curl -X PUT --silent -o /dev/null -w "%{http_code}"'` (printf `CURL_MAINTENANCE_GET_CMD="curl -X GET --silent %s ${CURL_URL}/v1/maintenance"` $adminCurlFlags)))) -}} +{{- $postStartSh := (list `#!/usr/bin/env bash` `# This code should be similar if not exactly the same as that found in the panda-operator, see` `# https://github.com/redpanda-data/redpanda/blob/e51d5b7f2ef76d5160ca01b8c7a8cf07593d29b6/src/go/k8s/pkg/resources/secret.go` `` `# path below should match the path defined on the statefulset` `source /var/lifecycle/common.sh` `` `postStartHook () {` ` set -x` `` ` touch /tmp/postStartHookStarted` `` ` until NODE_ID=$(${CURL_NODE_ID_CMD} | grep -o '\"node_id\":[^,}]*' | grep -o '[^: ]*$'); do` ` sleep 0.5` ` done` `` ` echo "Clearing maintenance mode on node ${NODE_ID}"` (printf ` CURL_MAINTENANCE_DELETE_CMD="${CURL_MAINTENANCE_DELETE_CMD_PREFIX} %s ${CURL_URL}/v1/brokers/${NODE_ID}/maintenance"` $adminCurlFlags) ` # a 400 here would mean not in maintenance mode` ` until [ "${status:-}" = '"200"' ] || [ "${status:-}" = '"400"' ]; do` ` status=$(${CURL_MAINTENANCE_DELETE_CMD})` ` sleep 0.5` ` done` `` ` touch /tmp/postStartHookFinished` `}` `` `postStartHook` `true`) -}} +{{- $_ := (set $secret.stringData "postStart.sh" (join "\n" $postStartSh)) -}} +{{- $preStopSh := (list `#!/usr/bin/env bash` `# This code should be similar if not exactly the same as that found in the panda-operator, see` `# https://github.com/redpanda-data/redpanda/blob/e51d5b7f2ef76d5160ca01b8c7a8cf07593d29b6/src/go/k8s/pkg/resources/secret.go` `` `touch /tmp/preStopHookStarted` `` `# path below should match the path defined on the statefulset` `source /var/lifecycle/common.sh` `` `set -x` `` `preStopHook () {` ` until NODE_ID=$(${CURL_NODE_ID_CMD} | grep -o '\"node_id\":[^,}]*' | grep -o '[^: ]*$'); do` ` sleep 0.5` ` done` `` ` echo "Setting maintenance mode on node ${NODE_ID}"` (printf ` CURL_MAINTENANCE_PUT_CMD="${CURL_MAINTENANCE_PUT_CMD_PREFIX} %s ${CURL_URL}/v1/brokers/${NODE_ID}/maintenance"` $adminCurlFlags) ` until [ "${status:-}" = '"200"' ]; do` ` status=$(${CURL_MAINTENANCE_PUT_CMD})` ` sleep 0.5` ` done` `` ` until [ "${finished:-}" = "true" ] || [ "${draining:-}" = "false" ]; do` ` res=$(${CURL_MAINTENANCE_GET_CMD})` ` finished=$(echo $res | grep -o '\"finished\":[^,}]*' | grep -o '[^: ]*$')` ` draining=$(echo $res | grep -o '\"draining\":[^,}]*' | grep -o '[^: ]*$')` ` sleep 0.5` ` done` `` ` touch /tmp/preStopHookFinished` `}`) -}} +{{- if (and (gt ($values.statefulset.replicas | int) (2 | int)) (not (get (fromJson (include "_shims.typeassertion" (dict "a" (list "bool" (dig "recovery_mode_enabled" false $values.config.node)) ))) "r"))) -}} +{{- $preStopSh = (concat (default (list ) $preStopSh) (list `preStopHook`)) -}} +{{- else -}} +{{- $preStopSh = (concat (default (list ) $preStopSh) (list `touch /tmp/preStopHookFinished` `echo "Not enough replicas or in recovery mode, cannot put a broker into maintenance mode."`)) -}} +{{- end -}} +{{- $preStopSh = (concat (default (list ) $preStopSh) (list `true`)) -}} +{{- $_ := (set $secret.stringData "preStop.sh" (join "\n" $preStopSh)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secret) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretSASLUsers" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (and (and (ne $values.auth.sasl.secretRef "") $values.auth.sasl.enabled) (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.auth.sasl.users) ))) "r") | int) (0 | int))) -}} +{{- $secret := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $values.auth.sasl.secretRef "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict ) )) -}} +{{- $usersTxt := (list ) -}} +{{- range $_, $user := $values.auth.sasl.users -}} +{{- if (empty $user.mechanism) -}} +{{- $usersTxt = (concat (default (list ) $usersTxt) (list (printf "%s:%s" $user.name $user.password))) -}} +{{- else -}} +{{- $usersTxt = (concat (default (list ) $usersTxt) (list (printf "%s:%s:%s" $user.name $user.password $user.mechanism))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $secret.stringData "users.txt" (join "\n" $usersTxt)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secret) | toJson -}} +{{- break -}} +{{- else -}}{{- if (and $values.auth.sasl.enabled (eq $values.auth.sasl.secretRef "")) -}} +{{- $_ := (fail "auth.sasl.secretRef cannot be empty when auth.sasl.enabled=true") -}} +{{- else -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretBootstrapUser" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.auth.sasl.enabled) (ne (toJson $values.auth.sasl.bootstrapUser.secretKeyRef) "null")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $secretName := (printf "%s-bootstrap-user" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- if $dot.Release.IsUpgrade -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.lookup" (dict "a" (list "v1" "Secret" $dot.Release.Namespace $secretName) ))) "r")) ))) "r") -}} +{{- $ok_6 := $tmp_tuple_1.T2 -}} +{{- $existing_5 := $tmp_tuple_1.T1 -}} +{{- if $ok_6 -}} +{{- $_is_returning = true -}} +{{- (dict "r" $existing_5) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $password := (randAlphaNum (32 | int)) -}} +{{- $userPassword := $values.auth.sasl.bootstrapUser.password -}} +{{- if (ne (toJson $userPassword) "null") -}} +{{- $password = $userPassword -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" $secretName "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict "password" $password ) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretConfigWatcher" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.statefulset.sideCars.configWatcher.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $bootstrapUser := (get (fromJson (include "redpanda.BootstrapUser.Username" (dict "a" (list $values.auth.sasl.bootstrapUser) ))) "r") -}} +{{- $sasl := $values.auth.sasl -}} +{{- $secret := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-config-watcher" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict ) )) -}} +{{- $saslUserSh := (coalesce nil) -}} +{{- $saslUserSh = (concat (default (list ) $saslUserSh) (list `#!/usr/bin/env bash` `` `trap 'error_handler $? $LINENO' ERR` `` `error_handler() {` ` echo "Error: ($1) occurred at line $2"` `}` `` `set -e` `` `# rpk cluster health can exit non-zero if it's unable to dial brokers. This` `# can happen for many reasons but we never want this script to crash as it` `# would take down yet another broker and make a bad situation worse.` `# Instead, just wait for the command to eventually exit zero.` `echo "Waiting for cluster to be ready"` `until rpk cluster health --watch --exit-when-healthy; do` ` echo "rpk cluster health failed. Waiting 5 seconds before trying again..."` ` sleep 5` `done`)) -}} +{{- if (and $sasl.enabled (ne $sasl.secretRef "")) -}} +{{- $saslUserSh = (concat (default (list ) $saslUserSh) (list `while true; do` ` echo "RUNNING: Monitoring and Updating SASL users"` ` USERS_DIR="/etc/secrets/users"` `` ` new_users_list(){` ` LIST=$1` ` NEW_USER=$2` ` if [[ -n "${LIST}" ]]; then` ` LIST="${NEW_USER},${LIST}"` ` else` ` LIST="${NEW_USER}"` ` fi` `` ` echo "${LIST}"` ` }` `` ` process_users() {` ` USERS_DIR=${1-"/etc/secrets/users"}` ` USERS_FILE=$(find ${USERS_DIR}/* -print)` (printf ` USERS_LIST="%s"` $bootstrapUser) ` READ_LIST_SUCCESS=0` ` # Read line by line, handle a missing EOL at the end of file` ` while read p || [ -n "$p" ] ; do` ` IFS=":" read -r USER_NAME PASSWORD MECHANISM <<< $p` ` # Do not process empty lines` ` if [ -z "$USER_NAME" ]; then` ` continue` ` fi` ` if [[ "${USER_NAME// /}" != "$USER_NAME" ]]; then` ` continue` ` fi` ` echo "Creating user ${USER_NAME}..."` (printf ` MECHANISM=${MECHANISM:-%s}` (dig "auth" "sasl" "mechanism" "SCRAM-SHA-512" $dot.Values.AsMap)) ` creation_result=$(rpk acl user create ${USER_NAME} -p ${PASSWORD} --mechanism ${MECHANISM} 2>&1) && creation_result_exit_code=$? || creation_result_exit_code=$? # On a non-success exit code` ` if [[ $creation_result_exit_code -ne 0 ]]; then` ` # Check if the stderr contains "User already exists"` ` # this error occurs when password has changed` ` if [[ $creation_result == *"User already exists"* ]]; then` ` echo "Update user ${USER_NAME}"` ` # we will try to update by first deleting` ` deletion_result=$(rpk acl user delete ${USER_NAME} 2>&1) && deletion_result_exit_code=$? || deletion_result_exit_code=$?` ` if [[ $deletion_result_exit_code -ne 0 ]]; then` ` echo "deletion of user ${USER_NAME} failed: ${deletion_result}"` ` READ_LIST_SUCCESS=1` ` break` ` fi` ` # Now we update the user` ` update_result=$(rpk acl user create ${USER_NAME} -p ${PASSWORD} --mechanism ${MECHANISM} 2>&1) && update_result_exit_code=$? || update_result_exit_code=$? # On a non-success exit code` ` if [[ $update_result_exit_code -ne 0 ]]; then` ` echo "updating user ${USER_NAME} failed: ${update_result}"` ` READ_LIST_SUCCESS=1` ` break` ` else` ` echo "Updated user ${USER_NAME}..."` ` USERS_LIST=$(new_users_list "${USERS_LIST}" "${USER_NAME}")` ` fi` ` else` ` # Another error occurred, so output the original message and exit code` ` echo "error creating user ${USER_NAME}: ${creation_result}"` ` READ_LIST_SUCCESS=1` ` break` ` fi` ` # On a success, the user was created so output that` ` else` ` echo "Created user ${USER_NAME}..."` ` USERS_LIST=$(new_users_list "${USERS_LIST}" "${USER_NAME}")` ` fi` ` done < $USERS_FILE` `` ` if [[ -n "${USERS_LIST}" && ${READ_LIST_SUCCESS} ]]; then` ` echo "Setting superusers configurations with users [${USERS_LIST}]"` ` superuser_result=$(rpk cluster config set superusers [${USERS_LIST}] 2>&1) && superuser_result_exit_code=$? || superuser_result_exit_code=$?` ` if [[ $superuser_result_exit_code -ne 0 ]]; then` ` echo "Setting superusers configurations failed: ${superuser_result}"` ` else` ` echo "Completed setting superusers configurations"` ` fi` ` fi` ` }` `` ` # before we do anything ensure we have the bootstrap user` ` echo "Ensuring bootstrap user ${RPK_USER}..."` ` creation_result=$(rpk acl user create ${RPK_USER} -p ${RPK_PASS} --mechanism ${RPK_SASL_MECHANISM} 2>&1) && creation_result_exit_code=$? || creation_result_exit_code=$? # On a non-success exit code` ` if [[ $creation_result_exit_code -ne 0 ]]; then` ` if [[ $creation_result == *"User already exists"* ]]; then` ` echo "Bootstrap user already created"` ` else` ` echo "error creating user ${RPK_USER}: ${creation_result}"` ` fi` ` fi` `` ` # first time processing` ` process_users $USERS_DIR` `` ` # subsequent changes detected here` ` # watching delete_self as documented in https://ahmet.im/blog/kubernetes-inotify/` ` USERS_FILE=$(find ${USERS_DIR}/* -print)` ` while RES=$(inotifywait -q -e delete_self ${USERS_FILE}); do` ` process_users $USERS_DIR` ` done` `done`)) -}} +{{- else -}} +{{- $saslUserSh = (concat (default (list ) $saslUserSh) (list `echo "Nothing to do. Sleeping..."` `sleep infinity`)) -}} +{{- end -}} +{{- $_ := (set $secret.stringData "sasl-user.sh" (join "\n" $saslUserSh)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secret) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretFSValidator" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.statefulset.initContainers.fsValidator.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $secret := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-fs-validator" (substr 0 (49 | int) (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r"))) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict ) )) -}} +{{- $_ := (set $secret.stringData "fsValidator.sh" `set -e +EXPECTED_FS_TYPE=$1 + +DATA_DIR="/var/lib/redpanda/data" +TEST_FILE="testfile" + +echo "checking data directory exist..." +if [ ! -d "${DATA_DIR}" ]; then + echo "data directory does not exists, exiting" + exit 1 +fi + +echo "checking filesystem type..." +FS_TYPE=$(df -T $DATA_DIR | tail -n +2 | awk '{print $2}') + +if [ "${FS_TYPE}" != "${EXPECTED_FS_TYPE}" ]; then + echo "file system found to be ${FS_TYPE} when expected ${EXPECTED_FS_TYPE}" + exit 1 +fi + +echo "checking if able to create a test file..." + +touch ${DATA_DIR}/${TEST_FILE} +result=$(touch ${DATA_DIR}/${TEST_FILE} 2> /dev/null; echo $?) +if [ "${result}" != "0" ]; then + echo "could not write testfile, may not have write permission" + exit 1 +fi + +echo "checking if able to delete a test file..." + +result=$(rm ${DATA_DIR}/${TEST_FILE} 2> /dev/null; echo $?) +if [ "${result}" != "0" ]; then + echo "could not delete testfile" + exit 1 +fi + +echo "passed"`) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secret) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretConfigurator" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $secret := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Secret" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%.51s-configurator" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "type" "Opaque" "stringData" (dict ) )) -}} +{{- $configuratorSh := (list ) -}} +{{- $configuratorSh = (concat (default (list ) $configuratorSh) (list `set -xe` `SERVICE_NAME=$1` `KUBERNETES_NODE_NAME=$2` `POD_ORDINAL=${SERVICE_NAME##*-}` "BROKER_INDEX=`expr $POD_ORDINAL + 1`" `` `CONFIG=/etc/redpanda/redpanda.yaml` `` `# Setup config files` `cp /tmp/base-config/redpanda.yaml "${CONFIG}"`)) -}} +{{- if (not (get (fromJson (include "redpanda.RedpandaAtLeast_22_3_0" (dict "a" (list $dot) ))) "r")) -}} +{{- $configuratorSh = (concat (default (list ) $configuratorSh) (list `` `# Configure bootstrap` `## Not used for Redpanda v22.3.0+` `rpk --config "${CONFIG}" redpanda config set redpanda.node_id "${POD_ORDINAL}"` `if [ "${POD_ORDINAL}" = "0" ]; then` ` rpk --config "${CONFIG}" redpanda config set redpanda.seed_servers '[]' --format yaml` `fi`)) -}} +{{- end -}} +{{- $kafkaSnippet := (get (fromJson (include "redpanda.secretConfiguratorKafkaConfig" (dict "a" (list $dot) ))) "r") -}} +{{- $configuratorSh = (concat (default (list ) $configuratorSh) (default (list ) $kafkaSnippet)) -}} +{{- $httpSnippet := (get (fromJson (include "redpanda.secretConfiguratorHTTPConfig" (dict "a" (list $dot) ))) "r") -}} +{{- $configuratorSh = (concat (default (list ) $configuratorSh) (default (list ) $httpSnippet)) -}} +{{- if (and (get (fromJson (include "redpanda.RedpandaAtLeast_22_3_0" (dict "a" (list $dot) ))) "r") $values.rackAwareness.enabled) -}} +{{- $configuratorSh = (concat (default (list ) $configuratorSh) (list `` `# Configure Rack Awareness` `set +x` (printf `RACK=$(curl --silent --cacert /run/secrets/kubernetes.io/serviceaccount/ca.crt --fail -H 'Authorization: Bearer '$(cat /run/secrets/kubernetes.io/serviceaccount/token) "https://${KUBERNETES_SERVICE_HOST}:${KUBERNETES_SERVICE_PORT_HTTPS}/api/v1/nodes/${KUBERNETES_NODE_NAME}?pretty=true" | grep %s | grep -v '\"key\":' | sed 's/.*": "\([^"]\+\).*/\1/')` (squote (quote $values.rackAwareness.nodeAnnotation))) `set -x` `rpk --config "$CONFIG" redpanda config set redpanda.rack "${RACK}"`)) -}} +{{- end -}} +{{- $_ := (set $secret.stringData "configurator.sh" (join "\n" $configuratorSh)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $secret) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.secretConfiguratorKafkaConfig" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $internalAdvertiseAddress := (printf "%s.%s" "${SERVICE_NAME}" (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) -}} +{{- $snippet := (coalesce nil) -}} +{{- $listenerName := "kafka" -}} +{{- $listenerAdvertisedName := $listenerName -}} +{{- $redpandaConfigPart := "redpanda" -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `LISTENER=%s` (quote (toJson (dict "name" "internal" "address" $internalAdvertiseAddress "port" ($values.listeners.kafka.port | int) )))) (printf `rpk redpanda config --config "$CONFIG" set %s.advertised_%s_api[0] "$LISTENER"` $redpandaConfigPart $listenerAdvertisedName))) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.listeners.kafka.external) ))) "r") | int) (0 | int)) -}} +{{- $externalCounter := (0 | int) -}} +{{- range $externalName, $externalVals := $values.listeners.kafka.external -}} +{{- $externalCounter = ((add $externalCounter (1 | int)) | int) -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `ADVERTISED_%s_ADDRESSES=()` (upper $listenerName)))) -}} +{{- range $_, $replicaIndex := (until (($values.statefulset.replicas | int) | int)) -}} +{{- $port := ($externalVals.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $externalVals.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $externalVals.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = (index $externalVals.advertisedPorts (0 | int)) -}} +{{- else -}} +{{- $port = (index $externalVals.advertisedPorts $replicaIndex) -}} +{{- end -}} +{{- end -}} +{{- $host := (get (fromJson (include "redpanda.advertisedHostJSON" (dict "a" (list $dot $externalName $port $replicaIndex) ))) "r") -}} +{{- $address := (toJson $host) -}} +{{- $prefixTemplate := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $externalVals.prefixTemplate "") ))) "r") -}} +{{- if (eq $prefixTemplate "") -}} +{{- $prefixTemplate = (default "" $values.external.prefixTemplate) -}} +{{- end -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `PREFIX_TEMPLATE=%s` (quote $prefixTemplate)) (printf `ADVERTISED_%s_ADDRESSES+=(%s)` (upper $listenerName) (quote $address)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `rpk redpanda config --config "$CONFIG" set %s.advertised_%s_api[%d] "${ADVERTISED_%s_ADDRESSES[$POD_ORDINAL]}"` $redpandaConfigPart $listenerAdvertisedName $externalCounter (upper $listenerName)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $snippet) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.secretConfiguratorHTTPConfig" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $internalAdvertiseAddress := (printf "%s.%s" "${SERVICE_NAME}" (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) -}} +{{- $snippet := (coalesce nil) -}} +{{- $listenerName := "http" -}} +{{- $listenerAdvertisedName := "pandaproxy" -}} +{{- $redpandaConfigPart := "pandaproxy" -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `LISTENER=%s` (quote (toJson (dict "name" "internal" "address" $internalAdvertiseAddress "port" ($values.listeners.http.port | int) )))) (printf `rpk redpanda config --config "$CONFIG" set %s.advertised_%s_api[0] "$LISTENER"` $redpandaConfigPart $listenerAdvertisedName))) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.listeners.http.external) ))) "r") | int) (0 | int)) -}} +{{- $externalCounter := (0 | int) -}} +{{- range $externalName, $externalVals := $values.listeners.http.external -}} +{{- $externalCounter = ((add $externalCounter (1 | int)) | int) -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `ADVERTISED_%s_ADDRESSES=()` (upper $listenerName)))) -}} +{{- range $_, $replicaIndex := (until (($values.statefulset.replicas | int) | int)) -}} +{{- $port := ($externalVals.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $externalVals.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $externalVals.advertisedPorts) ))) "r") | int) (1 | int)) -}} +{{- $port = (index $externalVals.advertisedPorts (0 | int)) -}} +{{- else -}} +{{- $port = (index $externalVals.advertisedPorts $replicaIndex) -}} +{{- end -}} +{{- end -}} +{{- $host := (get (fromJson (include "redpanda.advertisedHostJSON" (dict "a" (list $dot $externalName $port $replicaIndex) ))) "r") -}} +{{- $address := (toJson $host) -}} +{{- $prefixTemplate := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $externalVals.prefixTemplate "") ))) "r") -}} +{{- if (eq $prefixTemplate "") -}} +{{- $prefixTemplate = (default "" $values.external.prefixTemplate) -}} +{{- end -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `PREFIX_TEMPLATE=%s` (quote $prefixTemplate)) (printf `ADVERTISED_%s_ADDRESSES+=(%s)` (upper $listenerName) (quote $address)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $snippet = (concat (default (list ) $snippet) (list `` (printf `rpk redpanda config --config "$CONFIG" set %s.advertised_%s_api[%d] "${ADVERTISED_%s_ADDRESSES[$POD_ORDINAL]}"` $redpandaConfigPart $listenerAdvertisedName $externalCounter (upper $listenerName)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $snippet) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.adminTLSCurlFlags" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.admin.tls $values.tls) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" "") | toJson -}} +{{- break -}} +{{- end -}} +{{- if $values.listeners.admin.tls.requireClientAuth -}} +{{- $path := (printf "%s/%s-client" "/etc/tls/certs" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "--cacert %s/ca.crt --cert %s/tls.crt --key %s/tls.key" $path $path $path)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $path := (get (fromJson (include "redpanda.InternalTLS.ServerCAPath" (dict "a" (list $values.listeners.admin.tls $values.tls) ))) "r") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "--cacert %s" $path)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.externalAdvertiseAddress" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $eaa := "${SERVICE_NAME}" -}} +{{- $externalDomainTemplate := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.domain "") ))) "r") -}} +{{- $expanded := (tpl $externalDomainTemplate $dot) -}} +{{- if (not (empty $expanded)) -}} +{{- $eaa = (printf "%s.%s" "${SERVICE_NAME}" $expanded) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $eaa) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.advertisedHostJSON" -}} +{{- $dot := (index .a 0) -}} +{{- $externalName := (index .a 1) -}} +{{- $port := (index .a 2) -}} +{{- $replicaIndex := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $host := (dict "name" $externalName "address" (get (fromJson (include "redpanda.externalAdvertiseAddress" (dict "a" (list $dot) ))) "r") "port" $port ) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (0 | int)) -}} +{{- $address := "" -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (1 | int)) -}} +{{- $address = (index $values.external.addresses $replicaIndex) -}} +{{- else -}} +{{- $address = (index $values.external.addresses (0 | int)) -}} +{{- end -}} +{{- $domain_7 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.domain "") ))) "r") -}} +{{- if (ne $domain_7 "") -}} +{{- $host = (dict "name" $externalName "address" (printf "%s.%s" $address (tpl $domain_7 $dot)) "port" $port ) -}} +{{- else -}} +{{- $host = (dict "name" $externalName "address" $address "port" $port ) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $host) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.adminInternalHTTPProtocol" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.admin.tls $values.tls) ))) "r") -}} +{{- $_is_returning = true -}} +{{- (dict "r" "https") | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "http") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.adminInternalURL" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s://%s.%s.%s.svc.%s:%d" (get (fromJson (include "redpanda.adminInternalHTTPProtocol" (dict "a" (list $dot) ))) "r") `${SERVICE_NAME}` (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r") $dot.Release.Namespace (trimSuffix "." $values.clusterDomain) ($values.listeners.admin.port | int))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_service.internal.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_service.internal.go.tpl new file mode 100644 index 0000000000..0719ec5fa3 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_service.internal.go.tpl @@ -0,0 +1,38 @@ +{{- /* Generated from "service_internal.go" */ -}} + +{{- define "redpanda.MonitoringEnabledLabel" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "monitoring.redpanda.com/enabled" (printf "%t" $values.monitoring.enabled) )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ServiceInternal" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $ports := (list ) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "admin" "protocol" "TCP" "appProtocol" $values.listeners.admin.appProtocol "port" ($values.listeners.admin.port | int) "targetPort" ($values.listeners.admin.port | int) )))) -}} +{{- if $values.listeners.http.enabled -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "http" "protocol" "TCP" "port" ($values.listeners.http.port | int) "targetPort" ($values.listeners.http.port | int) )))) -}} +{{- end -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "kafka" "protocol" "TCP" "port" ($values.listeners.kafka.port | int) "targetPort" ($values.listeners.kafka.port | int) )))) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "rpc" "protocol" "TCP" "port" ($values.listeners.rpc.port | int) "targetPort" ($values.listeners.rpc.port | int) )))) -}} +{{- if $values.listeners.schemaRegistry.enabled -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" "schemaregistry" "protocol" "TCP" "port" ($values.listeners.schemaRegistry.port | int) "targetPort" ($values.listeners.schemaRegistry.port | int) )))) -}} +{{- end -}} +{{- $annotations := (dict ) -}} +{{- if (ne (toJson $values.service) "null") -}} +{{- $annotations = $values.service.internal.annotations -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Service" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (merge (dict ) (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.MonitoringEnabledLabel" (dict "a" (list $dot) ))) "r")) "annotations" $annotations )) "spec" (mustMergeOverwrite (dict ) (dict "type" "ClusterIP" "publishNotReadyAddresses" true "clusterIP" "None" "selector" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") "ports" $ports )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_service.loadbalancer.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_service.loadbalancer.go.tpl new file mode 100644 index 0000000000..bb34c583ed --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_service.loadbalancer.go.tpl @@ -0,0 +1,105 @@ +{{- /* Generated from "service.loadbalancer.go" */ -}} + +{{- define "redpanda.LoadBalancerServices" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.external.enabled) (not $values.external.service.enabled)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (ne $values.external.type "LoadBalancer") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $externalDNS := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.externalDns (mustMergeOverwrite (dict "enabled" false ) (dict ))) ))) "r") -}} +{{- $labels := (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") -}} +{{- $_ := (set $labels "repdanda.com/type" "loadbalancer") -}} +{{- $selector := (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") -}} +{{- $services := (coalesce nil) -}} +{{- $replicas := ($values.statefulset.replicas | int) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) (($values.statefulset.replicas | int)|int) (1|int) -}} +{{- $podname := (printf "%s-%d" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") $i) -}} +{{- $annotations := (dict ) -}} +{{- range $k, $v := $values.external.annotations -}} +{{- $_ := (set $annotations $k $v) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if $externalDNS.enabled -}} +{{- $prefix := $podname -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (0 | int)) -}} +{{- if (eq ((get (fromJson (include "_shims.len" (dict "a" (list $values.external.addresses) ))) "r") | int) (1 | int)) -}} +{{- $prefix = (index $values.external.addresses (0 | int)) -}} +{{- else -}} +{{- $prefix = (index $values.external.addresses $i) -}} +{{- end -}} +{{- end -}} +{{- $address := (printf "%s.%s" $prefix (tpl $values.external.domain $dot)) -}} +{{- $_ := (set $annotations "external-dns.alpha.kubernetes.io/hostname" $address) -}} +{{- end -}} +{{- $podSelector := (dict ) -}} +{{- range $k, $v := $selector -}} +{{- $_ := (set $podSelector $k $v) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $podSelector "statefulset.kubernetes.io/pod-name" $podname) -}} +{{- $ports := (coalesce nil) -}} +{{- range $name, $listener := $values.listeners.admin.external -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.enabled $values.external.enabled) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $fallbackPorts := (concat (default (list ) $listener.advertisedPorts) (list ($values.listeners.admin.port | int))) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "admin-%s" $name) "protocol" "TCP" "targetPort" ($listener.port | int) "port" ((get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.nodePort (index $fallbackPorts (0 | int))) ))) "r") | int) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.kafka.external -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.enabled $values.external.enabled) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $fallbackPorts := (concat (default (list ) $listener.advertisedPorts) (list ($listener.port | int))) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "kafka-%s" $name) "protocol" "TCP" "targetPort" ($listener.port | int) "port" ((get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.nodePort (index $fallbackPorts (0 | int))) ))) "r") | int) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.http.external -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.enabled $values.external.enabled) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $fallbackPorts := (concat (default (list ) $listener.advertisedPorts) (list ($listener.port | int))) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "http-%s" $name) "protocol" "TCP" "targetPort" ($listener.port | int) "port" ((get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.nodePort (index $fallbackPorts (0 | int))) ))) "r") | int) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.schemaRegistry.external -}} +{{- if (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.enabled $values.external.enabled) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $fallbackPorts := (concat (default (list ) $listener.advertisedPorts) (list ($listener.port | int))) -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "schema-%s" $name) "protocol" "TCP" "targetPort" ($listener.port | int) "port" ((get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $listener.nodePort (index $fallbackPorts (0 | int))) ))) "r") | int) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $svc := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Service" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "lb-%s" $podname) "namespace" $dot.Release.Namespace "labels" $labels "annotations" $annotations )) "spec" (mustMergeOverwrite (dict ) (dict "externalTrafficPolicy" "Local" "loadBalancerSourceRanges" $values.external.sourceRanges "ports" $ports "publishNotReadyAddresses" true "selector" $podSelector "sessionAffinity" "None" "type" "LoadBalancer" )) )) -}} +{{- $services = (concat (default (list ) $services) (list $svc)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $services) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_service.nodeport.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_service.nodeport.go.tpl new file mode 100644 index 0000000000..bc199951d7 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_service.nodeport.go.tpl @@ -0,0 +1,80 @@ +{{- /* Generated from "service.nodeport.go" */ -}} + +{{- define "redpanda.NodePortService" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.external.enabled) (not $values.external.service.enabled)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (ne $values.external.type "NodePort") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $ports := (coalesce nil) -}} +{{- range $name, $listener := $values.listeners.admin.external -}} +{{- if (not (get (fromJson (include "redpanda.AdminExternal.IsEnabled" (dict "a" (list $listener) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $nodePort := ($listener.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- $nodePort = (index $listener.advertisedPorts (0 | int)) -}} +{{- end -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "admin-%s" $name) "protocol" "TCP" "port" ($listener.port | int) "nodePort" $nodePort )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.kafka.external -}} +{{- if (not (get (fromJson (include "redpanda.KafkaExternal.IsEnabled" (dict "a" (list $listener) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $nodePort := ($listener.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- $nodePort = (index $listener.advertisedPorts (0 | int)) -}} +{{- end -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "kafka-%s" $name) "protocol" "TCP" "port" ($listener.port | int) "nodePort" $nodePort )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.http.external -}} +{{- if (not (get (fromJson (include "redpanda.HTTPExternal.IsEnabled" (dict "a" (list $listener) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $nodePort := ($listener.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- $nodePort = (index $listener.advertisedPorts (0 | int)) -}} +{{- end -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "http-%s" $name) "protocol" "TCP" "port" ($listener.port | int) "nodePort" $nodePort )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $name, $listener := $values.listeners.schemaRegistry.external -}} +{{- if (not (get (fromJson (include "redpanda.SchemaRegistryExternal.IsEnabled" (dict "a" (list $listener) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $nodePort := ($listener.port | int) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $listener.advertisedPorts) ))) "r") | int) (0 | int)) -}} +{{- $nodePort = (index $listener.advertisedPorts (0 | int)) -}} +{{- end -}} +{{- $ports = (concat (default (list ) $ports) (list (mustMergeOverwrite (dict "port" 0 "targetPort" 0 ) (dict "name" (printf "schema-%s" $name) "protocol" "TCP" "port" ($listener.port | int) "nodePort" $nodePort )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $annotations := $values.external.annotations -}} +{{- if (eq (toJson $annotations) "null") -}} +{{- $annotations = (dict ) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict ) "status" (dict "loadBalancer" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "Service" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (printf "%s-external" (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r")) "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $annotations )) "spec" (mustMergeOverwrite (dict ) (dict "externalTrafficPolicy" "Local" "ports" $ports "publishNotReadyAddresses" true "selector" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") "sessionAffinity" "None" "type" "NodePort" )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_serviceaccount.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_serviceaccount.go.tpl new file mode 100644 index 0000000000..82ec5be757 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_serviceaccount.go.tpl @@ -0,0 +1,18 @@ +{{- /* Generated from "serviceaccount.go" */ -}} + +{{- define "redpanda.ServiceAccount" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.serviceAccount.create) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "v1" "kind" "ServiceAccount" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") "annotations" $values.serviceAccount.annotations )) "automountServiceAccountToken" $values.serviceAccount.automountServiceAccountToken ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_servicemonitor.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_servicemonitor.go.tpl new file mode 100644 index 0000000000..7f5a621309 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_servicemonitor.go.tpl @@ -0,0 +1,26 @@ +{{- /* Generated from "servicemonitor.go" */ -}} + +{{- define "redpanda.ServiceMonitor" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.monitoring.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $endpoint := (mustMergeOverwrite (dict ) (dict "interval" $values.monitoring.scrapeInterval "path" "/public_metrics" "port" "admin" "enableHttp2" $values.monitoring.enableHttp2 "scheme" "http" )) -}} +{{- if (or (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $values.listeners.admin.tls $values.tls) ))) "r") (ne (toJson $values.monitoring.tlsConfig) "null")) -}} +{{- $_ := (set $endpoint "scheme" "https") -}} +{{- $_ := (set $endpoint "tlsConfig" $values.monitoring.tlsConfig) -}} +{{- if (eq (toJson $endpoint.tlsConfig) "null") -}} +{{- $_ := (set $endpoint "tlsConfig" (mustMergeOverwrite (dict "ca" (dict ) "cert" (dict ) ) (mustMergeOverwrite (dict "ca" (dict ) "cert" (dict ) ) (dict "insecureSkipVerify" true )) (dict ))) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "endpoints" (coalesce nil) "selector" (dict ) "namespaceSelector" (dict ) ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "monitoring.coreos.com/v1" "kind" "ServiceMonitor" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (merge (dict ) (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") $values.monitoring.labels) )) "spec" (mustMergeOverwrite (dict "endpoints" (coalesce nil) "selector" (dict ) "namespaceSelector" (dict ) ) (dict "endpoints" (list $endpoint) "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" (dict "monitoring.redpanda.com/enabled" "true" "app.kubernetes.io/name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") "app.kubernetes.io/instance" $dot.Release.Name ) )) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_shims.tpl b/charts/redpanda/redpanda/5.9.10/templates/_shims.tpl new file mode 100644 index 0000000000..c16b6d1788 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_shims.tpl @@ -0,0 +1,339 @@ +{{- /* Generated from "bootstrap.go" */ -}} + +{{- define "_shims.typetest" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs $typ $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.typeassertion" -}} +{{- $typ := (index .a 0) -}} +{{- $value := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not (typeIs $typ $value)) -}} +{{- $_ := (fail (printf "expected type of %q got: %T" $typ $value)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $value) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.dicttest" -}} +{{- $m := (index .a 0) -}} +{{- $key := (index .a 1) -}} +{{- $zero := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (hasKey $m $key) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (index $m $key) true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $zero false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.compact" -}} +{{- $args := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $out := (dict ) -}} +{{- range $i, $e := $args -}} +{{- $_ := (set $out (printf "T%d" ((add (1 | int) $i) | int)) $e) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $out) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.deref" -}} +{{- $ptr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $ptr) "null") -}} +{{- $_ := (fail "nil dereference") -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.len" -}} +{{- $m := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $m) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (0 | int)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (len $m)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Deref" -}} +{{- $ptr := (index .a 0) -}} +{{- $def := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $ptr) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ptr) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $def) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.ptr_Equal" -}} +{{- $a := (index .a 0) -}} +{{- $b := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (and (eq (toJson $a) "null") (eq (toJson $b) "null")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (eq $a $b)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.lookup" -}} +{{- $apiVersion := (index .a 0) -}} +{{- $kind := (index .a 1) -}} +{{- $namespace := (index .a 2) -}} +{{- $name := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (lookup $apiVersion $kind $namespace $name) -}} +{{- if (empty $result) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (coalesce nil) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $result true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asnumeric" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int64" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (typeIs "int" $value) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.asintegral" -}} +{{- $value := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (or (typeIs "int64" $value) (typeIs "int" $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (and (typeIs "float64" $value) (eq (floor $value) $value)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $value true)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (0 | int) false)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.parseResource" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (typeIs "float64" $repr) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (float64 $repr) 1.0)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (not (typeIs "string" $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity expected string or float64 got: %T (%v)" $repr $repr)) -}} +{{- end -}} +{{- if (not (regexMatch `^[0-9]+(\.[0-9]{0,6})?(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$` $repr)) -}} +{{- $_ := (fail (printf "invalid Quantity: %q" $repr)) -}} +{{- end -}} +{{- $reprStr := (toString $repr) -}} +{{- $unit := (regexFind "(k|m|M|G|T|P|Ki|Mi|Gi|Ti|Pi)$" $repr) -}} +{{- $numeric := (float64 (substr (0 | int) ((sub ((get (fromJson (include "_shims.len" (dict "a" (list $reprStr) ))) "r") | int) ((get (fromJson (include "_shims.len" (dict "a" (list $unit) ))) "r") | int)) | int) $reprStr)) -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list (dict "" 1.0 "m" 0.001 "k" (1000 | int) "M" (1000000 | int) "G" (1000000000 | int) "T" (1000000000000 | int) "P" (1000000000000000 | int) "Ki" (1024 | int) "Mi" (1048576 | int) "Gi" (1073741824 | int) "Ti" (1099511627776 | int) "Pi" (1125899906842624 | int) ) $unit (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_1.T2 -}} +{{- $scale := ($tmp_tuple_1.T1 | float64) -}} +{{- if (not $ok) -}} +{{- $_ := (fail (printf "unknown unit: %q" $unit)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $numeric $scale)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MustParse" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_2.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_2.T1 | float64) -}} +{{- $strs := (list "" "m" "k" "M" "G" "T" "P" "Ki" "Mi" "Gi" "Ti" "Pi") -}} +{{- $scales := (list 1.0 0.001 (1000 | int) (1000000 | int) (1000000000 | int) (1000000000000 | int) (1000000000000000 | int) (1024 | int) (1048576 | int) (1073741824 | int) (1099511627776 | int) (1125899906842624 | int)) -}} +{{- $idx := -1 -}} +{{- range $i, $s := $scales -}} +{{- if (eq ($s | float64) ($scale | float64)) -}} +{{- $idx = $i -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (eq $idx -1) -}} +{{- $_ := (fail (printf "unknown scale: %v" $scale)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s%s" (toString $numeric) (index $strs $idx))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_Value" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_3.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_3.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf $numeric $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.resource_MilliValue" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.parseResource" (dict "a" (list $repr) ))) "r")) ))) "r") -}} +{{- $scale := ($tmp_tuple_4.T2 | float64) -}} +{{- $numeric := ($tmp_tuple_4.T1 | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (int64 (ceil ((mulf ((mulf $numeric 1000.0) | float64) $scale) | float64)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.time_ParseDuration" -}} +{{- $repr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $unitMap := (dict "s" (1000000000 | int64) "m" (60000000000 | int64) "h" (3600000000000 | int64) ) -}} +{{- $original := $repr -}} +{{- $value := ((0 | int64) | int64) -}} +{{- if (eq $repr "") -}} +{{- $_ := (fail (printf "invalid Duration: %q" $original)) -}} +{{- end -}} +{{- if (eq $repr "0") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (0 | int64)) | toJson -}} +{{- break -}} +{{- end -}} +{{- range $_, $_ := (list (0 | int) (0 | int) (0 | int)) -}} +{{- if (eq $repr "") -}} +{{- break -}} +{{- end -}} +{{- $n := (regexFind `^\d+` $repr) -}} +{{- if (eq $n "") -}} +{{- $_ := (fail (printf "invalid Duration: %q" $original)) -}} +{{- end -}} +{{- $repr = (substr ((get (fromJson (include "_shims.len" (dict "a" (list $n) ))) "r") | int) -1 $repr) -}} +{{- $unit := (regexFind `^(h|m|s)` $repr) -}} +{{- if (eq $unit "") -}} +{{- $_ := (fail (printf "invalid Duration: %q" $original)) -}} +{{- end -}} +{{- $repr = (substr ((get (fromJson (include "_shims.len" (dict "a" (list $unit) ))) "r") | int) -1 $repr) -}} +{{- $value = ((add $value (((mul (int64 $n) (index $unitMap $unit)) | int64))) | int64) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $value) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.time_Duration_String" -}} +{{- $dur := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (duration ((div $dur (1000000000 | int64)) | int64))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "_shims.render-manifest" -}} +{{- $tpl := (index . 0) -}} +{{- $dot := (index . 1) -}} +{{- $manifests := (get ((include $tpl (dict "a" (list $dot))) | fromJson) "r") -}} +{{- if not (typeIs "[]interface {}" $manifests) -}} +{{- $manifests = (list $manifests) -}} +{{- end -}} +{{- range $_, $manifest := $manifests -}} +{{- if ne (toJson $manifest) "null" }} +--- +{{toYaml (unset (unset $manifest "status") "creationTimestamp")}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/_statefulset.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_statefulset.go.tpl new file mode 100644 index 0000000000..6c1f9bbaf4 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_statefulset.go.tpl @@ -0,0 +1,777 @@ +{{- /* Generated from "statefulset.go" */ -}} + +{{- define "redpanda.statefulSetRedpandaEnv" -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "SERVICE_NAME" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "fieldPath" "metadata.name" )) )) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "POD_IP" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "fieldPath" "status.podIP" )) )) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "HOST_IP" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "fieldPath" "status.hostIP" )) )) )))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetPodLabelsSelector" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if $dot.Release.IsUpgrade -}} +{{- $tmp_tuple_1 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.lookup" (dict "a" (list "apps/v1" "StatefulSet" $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) ))) "r")) ))) "r") -}} +{{- $ok_2 := $tmp_tuple_1.T2 -}} +{{- $existing_1 := $tmp_tuple_1.T1 -}} +{{- if (and $ok_2 (gt ((get (fromJson (include "_shims.len" (dict "a" (list $existing_1.spec.selector.matchLabels) ))) "r") | int) (0 | int))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $existing_1.spec.selector.matchLabels) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $values := $dot.Values.AsMap -}} +{{- $additionalSelectorLabels := (dict ) -}} +{{- if (ne (toJson $values.statefulset.additionalSelectorLabels) "null") -}} +{{- $additionalSelectorLabels = $values.statefulset.additionalSelectorLabels -}} +{{- end -}} +{{- $component := (printf "%s-statefulset" (trimSuffix "-" (trunc (51 | int) (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r")))) -}} +{{- $defaults := (dict "app.kubernetes.io/component" $component "app.kubernetes.io/instance" $dot.Release.Name "app.kubernetes.io/name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") ) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $additionalSelectorLabels $defaults)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetPodLabels" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if $dot.Release.IsUpgrade -}} +{{- $tmp_tuple_2 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.lookup" (dict "a" (list "apps/v1" "StatefulSet" $dot.Release.Namespace (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) ))) "r")) ))) "r") -}} +{{- $ok_4 := $tmp_tuple_2.T2 -}} +{{- $existing_3 := $tmp_tuple_2.T1 -}} +{{- if (and $ok_4 (gt ((get (fromJson (include "_shims.len" (dict "a" (list $existing_3.spec.template.metadata.labels) ))) "r") | int) (0 | int))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $existing_3.spec.template.metadata.labels) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} +{{- $values := $dot.Values.AsMap -}} +{{- $statefulSetLabels := (dict ) -}} +{{- if (ne (toJson $values.statefulset.podTemplate.labels) "null") -}} +{{- $statefulSetLabels = $values.statefulset.podTemplate.labels -}} +{{- end -}} +{{- $defaults := (dict "redpanda.com/poddisruptionbudget" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") ) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $statefulSetLabels (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") $defaults (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetPodAnnotations" -}} +{{- $dot := (index .a 0) -}} +{{- $configMapChecksum := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $configMapChecksumAnnotation := (dict "config.redpanda.com/checksum" $configMapChecksum ) -}} +{{- if (ne (toJson $values.statefulset.podTemplate.annotations) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $values.statefulset.podTemplate.annotations $configMapChecksumAnnotation)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (merge (dict ) $values.statefulset.annotations $configMapChecksumAnnotation)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $fullname := (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") -}} +{{- $volumes := (get (fromJson (include "redpanda.CommonVolumes" (dict "a" (list $dot) ))) "r") -}} +{{- $values := $dot.Values.AsMap -}} +{{- $volumes = (concat (default (list ) $volumes) (default (list ) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (printf "%.50s-sts-lifecycle" $fullname) "defaultMode" (0o775 | int) )) )) (dict "name" "lifecycle-scripts" )) (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" $fullname )) (dict )) )) (dict "name" "base-config" )) (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "emptyDir" (mustMergeOverwrite (dict ) (dict )) )) (dict "name" "config" )) (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (printf "%.51s-configurator" $fullname) "defaultMode" (0o775 | int) )) )) (dict "name" (printf "%.51s-configurator" $fullname) )) (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (printf "%s-config-watcher" $fullname) "defaultMode" (0o775 | int) )) )) (dict "name" (printf "%s-config-watcher" $fullname) ))))) -}} +{{- if $values.statefulset.initContainers.fsValidator.enabled -}} +{{- $volumes = (concat (default (list ) $volumes) (list (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (dict "secretName" (printf "%.49s-fs-validator" $fullname) "defaultMode" (0o775 | int) )) )) (dict "name" (printf "%.49s-fs-validator" $fullname) )))) -}} +{{- end -}} +{{- $vol_5 := (get (fromJson (include "redpanda.Listeners.TrustStoreVolume" (dict "a" (list $values.listeners $values.tls) ))) "r") -}} +{{- if (ne (toJson $vol_5) "null") -}} +{{- $volumes = (concat (default (list ) $volumes) (list $vol_5)) -}} +{{- end -}} +{{- $volumes = (concat (default (list ) $volumes) (default (list ) (get (fromJson (include "redpanda.templateToVolumes" (dict "a" (list $dot $values.statefulset.extraVolumes) ))) "r"))) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (get (fromJson (include "redpanda.statefulSetVolumeDataDir" (dict "a" (list $dot) ))) "r"))) -}} +{{- $v_6 := (get (fromJson (include "redpanda.statefulSetVolumeTieredStorageDir" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $v_6) "null") -}} +{{- $volumes = (concat (default (list ) $volumes) (list $v_6)) -}} +{{- end -}} +{{- if (and (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.serviceAccount.automountServiceAccountToken false) ))) "r")) ((or ((and (and $values.rbac.enabled $values.statefulset.sideCars.controllers.enabled) $values.statefulset.sideCars.controllers.createRBAC)) $values.rackAwareness.enabled))) -}} +{{- $foundK8STokenVolume := false -}} +{{- range $_, $v := $volumes -}} +{{- if (hasPrefix $v.name (printf "%s%s" "kube-api-access" "-")) -}} +{{- $foundK8STokenVolume = true -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (not $foundK8STokenVolume) -}} +{{- $volumes = (concat (default (list ) $volumes) (list (get (fromJson (include "redpanda.kubeTokenAPIVolume" (dict "a" (list "kube-api-access") ))) "r"))) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $volumes) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.kubeTokenAPIVolume" -}} +{{- $name := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "projected" (mustMergeOverwrite (dict "sources" (coalesce nil) ) (dict "defaultMode" (420 | int) "sources" (list (mustMergeOverwrite (dict ) (dict "serviceAccountToken" (mustMergeOverwrite (dict "path" "" ) (dict "path" "token" "expirationSeconds" ((3607 | int) | int64) )) )) (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" "kube-root-ca.crt" )) (dict "items" (list (mustMergeOverwrite (dict "key" "" "path" "" ) (dict "key" "ca.crt" "path" "ca.crt" ))) )) )) (mustMergeOverwrite (dict ) (dict "downwardAPI" (mustMergeOverwrite (dict ) (dict "items" (list (mustMergeOverwrite (dict "path" "" ) (dict "path" "namespace" "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "apiVersion" "v1" "fieldPath" "metadata.namespace" )) ))) )) ))) )) )) (dict "name" $name ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetVolumeDataDir" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $datadirSource := (mustMergeOverwrite (dict ) (dict "emptyDir" (mustMergeOverwrite (dict ) (dict )) )) -}} +{{- if $values.storage.persistentVolume.enabled -}} +{{- $datadirSource = (mustMergeOverwrite (dict ) (dict "persistentVolumeClaim" (mustMergeOverwrite (dict "claimName" "" ) (dict "claimName" "datadir" )) )) -}} +{{- else -}}{{- if (ne $values.storage.hostPath "") -}} +{{- $datadirSource = (mustMergeOverwrite (dict ) (dict "hostPath" (mustMergeOverwrite (dict "path" "" ) (dict "path" $values.storage.hostPath )) )) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" ) $datadirSource (dict "name" "datadir" ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetVolumeTieredStorageDir" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (get (fromJson (include "redpanda.Storage.IsTieredStorageEnabled" (dict "a" (list $values.storage) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $tieredType := (get (fromJson (include "redpanda.Storage.TieredMountType" (dict "a" (list $values.storage) ))) "r") -}} +{{- if (or (eq $tieredType "none") (eq $tieredType "persistentVolume")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (eq $tieredType "hostPath") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "hostPath" (mustMergeOverwrite (dict "path" "" ) (dict "path" (get (fromJson (include "redpanda.Storage.GetTieredStorageHostPath" (dict "a" (list $values.storage) ))) "r") )) )) (dict "name" "tiered-storage-dir" ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "emptyDir" (mustMergeOverwrite (dict ) (dict "sizeLimit" (get (fromJson (include "redpanda.TieredStorageConfig.CloudStorageCacheSize" (dict "a" (list (deepCopy (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r"))) ))) "r") )) )) (dict "name" "tiered-storage-dir" ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetVolumeMounts" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $mounts := (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r") -}} +{{- $values := $dot.Values.AsMap -}} +{{- $mounts = (concat (default (list ) $mounts) (default (list ) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "config" "mountPath" "/etc/redpanda" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/tmp/base-config" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "lifecycle-scripts" "mountPath" "/var/lifecycle" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "datadir" "mountPath" "/var/lib/redpanda/data" ))))) -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list (get (fromJson (include "redpanda.Listeners.TrustStores" (dict "a" (list $values.listeners $values.tls) ))) "r")) ))) "r") | int) (0 | int)) -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "truststores" "mountPath" "/etc/truststores" "readOnly" true )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $mounts) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetInitContainers" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $containers := (coalesce nil) -}} +{{- $c_7 := (get (fromJson (include "redpanda.statefulSetInitContainerTuning" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_7) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_7)) -}} +{{- end -}} +{{- $c_8 := (get (fromJson (include "redpanda.statefulSetInitContainerSetDataDirOwnership" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_8) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_8)) -}} +{{- end -}} +{{- $c_9 := (get (fromJson (include "redpanda.statefulSetInitContainerFSValidator" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_9) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_9)) -}} +{{- end -}} +{{- $c_10 := (get (fromJson (include "redpanda.statefulSetInitContainerSetTieredStorageCacheDirOwnership" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_10) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_10)) -}} +{{- end -}} +{{- $containers = (concat (default (list ) $containers) (list (get (fromJson (include "redpanda.statefulSetInitContainerConfigurator" (dict "a" (list $dot) ))) "r"))) -}} +{{- $containers = (concat (default (list ) $containers) (list (get (fromJson (include "redpanda.bootstrapYamlTemplater" (dict "a" (list $dot) ))) "r"))) -}} +{{- $containers = (concat (default (list ) $containers) (default (list ) (get (fromJson (include "redpanda.templateToContainers" (dict "a" (list $dot $values.statefulset.initContainers.extraInitContainers) ))) "r"))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $containers) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetInitContainerTuning" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.tuning.tune_aio_events) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "tuning" "image" (printf "%s:%s" $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "command" (list `/bin/bash` `-c` `rpk redpanda tune all`) "securityContext" (mustMergeOverwrite (dict ) (dict "capabilities" (mustMergeOverwrite (dict ) (dict "add" (list `SYS_RESOURCE`) )) "privileged" true "runAsUser" ((0 | int64) | int64) "runAsGroup" ((0 | int64) | int64) )) "volumeMounts" (concat (default (list ) (concat (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.initContainers.tuning.extraVolumeMounts) ))) "r")))) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/etc/redpanda" )))) "resources" $values.statefulset.initContainers.tuning.resources ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetInitContainerSetDataDirOwnership" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.statefulset.initContainers.setDataDirOwnership.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "redpanda.securityContextUidGid" (dict "a" (list $dot "set-datadir-ownership") ))) "r")) ))) "r") -}} +{{- $gid := ($tmp_tuple_3.T2 | int64) -}} +{{- $uid := ($tmp_tuple_3.T1 | int64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "set-datadir-ownership" "image" (printf "%s:%s" $values.statefulset.initContainerImage.repository $values.statefulset.initContainerImage.tag) "command" (list `/bin/sh` `-c` (printf `chown %d:%d -R /var/lib/redpanda/data` $uid $gid)) "volumeMounts" (concat (default (list ) (concat (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.initContainers.setDataDirOwnership.extraVolumeMounts) ))) "r")))) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" `datadir` "mountPath" `/var/lib/redpanda/data` )))) "resources" $values.statefulset.initContainers.setDataDirOwnership.resources ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.securityContextUidGid" -}} +{{- $dot := (index .a 0) -}} +{{- $containerName := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $uid := $values.statefulset.securityContext.runAsUser -}} +{{- if (and (ne (toJson $values.statefulset.podSecurityContext) "null") (ne (toJson $values.statefulset.podSecurityContext.runAsUser) "null")) -}} +{{- $uid = $values.statefulset.podSecurityContext.runAsUser -}} +{{- end -}} +{{- if (eq (toJson $uid) "null") -}} +{{- $_ := (fail (printf `%s container requires runAsUser to be specified` $containerName)) -}} +{{- end -}} +{{- $gid := $values.statefulset.securityContext.fsGroup -}} +{{- if (and (ne (toJson $values.statefulset.podSecurityContext) "null") (ne (toJson $values.statefulset.podSecurityContext.fsGroup) "null")) -}} +{{- $gid = $values.statefulset.podSecurityContext.fsGroup -}} +{{- end -}} +{{- if (eq (toJson $gid) "null") -}} +{{- $_ := (fail (printf `%s container requires fsGroup to be specified` $containerName)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list $uid $gid)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetInitContainerFSValidator" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.statefulset.initContainers.fsValidator.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "fs-validator" "image" (printf "%s:%s" $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "command" (list `/bin/sh`) "args" (list `-c` (printf `trap "exit 0" TERM; exec /etc/secrets/fs-validator/scripts/fsValidator.sh %s & wait $!` $values.statefulset.initContainers.fsValidator.expectedFS)) "securityContext" (get (fromJson (include "redpanda.ContainerSecurityContext" (dict "a" (list $dot) ))) "r") "volumeMounts" (concat (default (list ) (concat (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.initContainers.fsValidator.extraVolumeMounts) ))) "r")))) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf `%.49s-fs-validator` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "mountPath" `/etc/secrets/fs-validator/scripts/` )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" `datadir` "mountPath" `/var/lib/redpanda/data` )))) "resources" $values.statefulset.initContainers.fsValidator.resources ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetInitContainerSetTieredStorageCacheDirOwnership" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not (get (fromJson (include "redpanda.Storage.IsTieredStorageEnabled" (dict "a" (list $values.storage) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "redpanda.securityContextUidGid" (dict "a" (list $dot "set-tiered-storage-cache-dir-ownership") ))) "r")) ))) "r") -}} +{{- $gid := ($tmp_tuple_4.T2 | int64) -}} +{{- $uid := ($tmp_tuple_4.T1 | int64) -}} +{{- $cacheDir := (get (fromJson (include "redpanda.Storage.TieredCacheDirectory" (dict "a" (list $values.storage $dot) ))) "r") -}} +{{- $mounts := (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r") -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "datadir" "mountPath" "/var/lib/redpanda/data" )))) -}} +{{- if (ne (get (fromJson (include "redpanda.Storage.TieredMountType" (dict "a" (list $values.storage) ))) "r") "none") -}} +{{- $name := "tiered-storage-dir" -}} +{{- if (and (ne (toJson $values.storage.persistentVolume) "null") (ne $values.storage.persistentVolume.nameOverwrite "")) -}} +{{- $name = $values.storage.persistentVolume.nameOverwrite -}} +{{- end -}} +{{- $mounts = (concat (default (list ) $mounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" $name "mountPath" $cacheDir )))) -}} +{{- end -}} +{{- $mounts = (concat (default (list ) $mounts) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.initContainers.setTieredStorageCacheDirOwnership.extraVolumeMounts) ))) "r"))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" `set-tiered-storage-cache-dir-ownership` "image" (printf `%s:%s` $values.statefulset.initContainerImage.repository $values.statefulset.initContainerImage.tag) "command" (list `/bin/sh` `-c` (printf `mkdir -p %s; chown %d:%d -R %s` $cacheDir $uid $gid $cacheDir)) "volumeMounts" $mounts "resources" $values.statefulset.initContainers.setTieredStorageCacheDirOwnership.resources ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetInitContainerConfigurator" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $volMounts := (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r") -}} +{{- $volMounts = (concat (default (list ) $volMounts) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.initContainers.configurator.extraVolumeMounts) ))) "r"))) -}} +{{- $volMounts = (concat (default (list ) $volMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "config" "mountPath" "/etc/redpanda" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "base-config" "mountPath" "/tmp/base-config" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf `%.51s-configurator` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "mountPath" "/etc/secrets/configurator/scripts/" )))) -}} +{{- if (and (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.serviceAccount.automountServiceAccountToken false) ))) "r")) $values.rackAwareness.enabled) -}} +{{- $mountName := "kube-api-access" -}} +{{- range $_, $vol := (get (fromJson (include "redpanda.StatefulSetVolumes" (dict "a" (list $dot) ))) "r") -}} +{{- if (hasPrefix $vol.name (printf "%s%s" "kube-api-access" "-")) -}} +{{- $mountName = $vol.name -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $volMounts = (concat (default (list ) $volMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" $mountName "readOnly" true "mountPath" "/var/run/secrets/kubernetes.io/serviceaccount" )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" (printf `%.51s-configurator` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r")) "image" (printf `%s:%s` $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "command" (list `/bin/bash` `-c` `trap "exit 0" TERM; exec $CONFIGURATOR_SCRIPT "${SERVICE_NAME}" "${KUBERNETES_NODE_NAME}" & wait $!`) "env" (get (fromJson (include "redpanda.rpkEnvVars" (dict "a" (list $dot (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "CONFIGURATOR_SCRIPT" "value" "/etc/secrets/configurator/scripts/configurator.sh" )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "SERVICE_NAME" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "fieldPath" "metadata.name" )) "resourceFieldRef" (coalesce nil) "configMapKeyRef" (coalesce nil) "secretKeyRef" (coalesce nil) )) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "KUBERNETES_NODE_NAME" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "fieldPath" "spec.nodeName" )) )) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "HOST_IP_ADDRESS" "valueFrom" (mustMergeOverwrite (dict ) (dict "fieldRef" (mustMergeOverwrite (dict "fieldPath" "" ) (dict "apiVersion" "v1" "fieldPath" "status.hostIP" )) )) )))) ))) "r") "securityContext" (get (fromJson (include "redpanda.ContainerSecurityContext" (dict "a" (list $dot) ))) "r") "volumeMounts" $volMounts "resources" $values.statefulset.initContainers.configurator.resources ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSetContainers" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $containers := (coalesce nil) -}} +{{- $containers = (concat (default (list ) $containers) (list (get (fromJson (include "redpanda.statefulSetContainerRedpanda" (dict "a" (list $dot) ))) "r"))) -}} +{{- $c_11 := (get (fromJson (include "redpanda.statefulSetContainerConfigWatcher" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_11) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_11)) -}} +{{- end -}} +{{- $c_12 := (get (fromJson (include "redpanda.statefulSetContainerControllers" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $c_12) "null") -}} +{{- $containers = (concat (default (list ) $containers) (list $c_12)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $containers) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.wrapLifecycleHook" -}} +{{- $hook := (index .a 0) -}} +{{- $timeoutSeconds := (index .a 1) -}} +{{- $cmd := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $wrapped := (join " " $cmd) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list "bash" "-c" (printf "timeout -v %d %s 2>&1 | sed \"s/^/lifecycle-hook %s $(date): /\" | tee /proc/1/fd/1; true" $timeoutSeconds $wrapped $hook))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetContainerRedpanda" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $internalAdvertiseAddress := (printf "%s.%s" "$(SERVICE_NAME)" (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r")) -}} +{{- $container := (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") "image" (printf `%s:%s` $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "env" (get (fromJson (include "redpanda.bootstrapEnvVars" (dict "a" (list $dot (get (fromJson (include "redpanda.statefulSetRedpandaEnv" (dict "a" (list ) ))) "r")) ))) "r") "lifecycle" (mustMergeOverwrite (dict ) (dict "postStart" (mustMergeOverwrite (dict ) (dict "exec" (mustMergeOverwrite (dict ) (dict "command" (get (fromJson (include "redpanda.wrapLifecycleHook" (dict "a" (list "post-start" ((div ($values.statefulset.terminationGracePeriodSeconds | int64) (2 | int64)) | int64) (list "bash" "-x" "/var/lifecycle/postStart.sh")) ))) "r") )) )) "preStop" (mustMergeOverwrite (dict ) (dict "exec" (mustMergeOverwrite (dict ) (dict "command" (get (fromJson (include "redpanda.wrapLifecycleHook" (dict "a" (list "pre-stop" ((div ($values.statefulset.terminationGracePeriodSeconds | int64) (2 | int64)) | int64) (list "bash" "-x" "/var/lifecycle/preStop.sh")) ))) "r") )) )) )) "startupProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "exec" (mustMergeOverwrite (dict ) (dict "command" (list `/bin/sh` `-c` (join "\n" (list `set -e` (printf `RESULT=$(curl --silent --fail -k -m 5 %s "%s://%s/v1/status/ready")` (get (fromJson (include "redpanda.adminTLSCurlFlags" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.adminInternalHTTPProtocol" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.adminApiURLs" (dict "a" (list $dot) ))) "r")) `echo $RESULT` `echo $RESULT | grep ready` ``))) )) )) (dict "initialDelaySeconds" ($values.statefulset.startupProbe.initialDelaySeconds | int) "periodSeconds" ($values.statefulset.startupProbe.periodSeconds | int) "failureThreshold" ($values.statefulset.startupProbe.failureThreshold | int) )) "livenessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "exec" (mustMergeOverwrite (dict ) (dict "command" (list `/bin/sh` `-c` (printf `curl --silent --fail -k -m 5 %s "%s://%s/v1/status/ready"` (get (fromJson (include "redpanda.adminTLSCurlFlags" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.adminInternalHTTPProtocol" (dict "a" (list $dot) ))) "r") (get (fromJson (include "redpanda.adminApiURLs" (dict "a" (list $dot) ))) "r"))) )) )) (dict "initialDelaySeconds" ($values.statefulset.livenessProbe.initialDelaySeconds | int) "periodSeconds" ($values.statefulset.livenessProbe.periodSeconds | int) "failureThreshold" ($values.statefulset.livenessProbe.failureThreshold | int) )) "command" (list `rpk` `redpanda` `start` (printf `--advertise-rpc-addr=%s:%d` $internalAdvertiseAddress ($values.listeners.rpc.port | int))) "volumeMounts" (concat (default (list ) (get (fromJson (include "redpanda.StatefulSetVolumeMounts" (dict "a" (list $dot) ))) "r")) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.extraVolumeMounts) ))) "r"))) "securityContext" (get (fromJson (include "redpanda.ContainerSecurityContext" (dict "a" (list $dot) ))) "r") "resources" (mustMergeOverwrite (dict ) (dict )) )) -}} +{{- if (not (get (fromJson (include "_shims.typeassertion" (dict "a" (list "bool" (dig `recovery_mode_enabled` false $values.config.node)) ))) "r")) -}} +{{- $_ := (set $container "readinessProbe" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "exec" (mustMergeOverwrite (dict ) (dict "command" (list `/bin/sh` `-c` (join "\n" (list `set -x` `RESULT=$(rpk cluster health)` `echo $RESULT` `echo $RESULT | grep 'Healthy:.*true'` ``))) )) )) (dict "initialDelaySeconds" ($values.statefulset.readinessProbe.initialDelaySeconds | int) "timeoutSeconds" ($values.statefulset.readinessProbe.timeoutSeconds | int) "periodSeconds" ($values.statefulset.readinessProbe.periodSeconds | int) "successThreshold" ($values.statefulset.readinessProbe.successThreshold | int) "failureThreshold" ($values.statefulset.readinessProbe.failureThreshold | int) ))) -}} +{{- end -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "admin" "containerPort" ($values.listeners.admin.port | int) ))))) -}} +{{- range $externalName, $external := $values.listeners.admin.external -}} +{{- if (get (fromJson (include "redpanda.AdminExternal.IsEnabled" (dict "a" (list $external) ))) "r") -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" (printf "admin-%.8s" (lower $externalName)) "containerPort" ($external.port | int) ))))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "http" "containerPort" ($values.listeners.http.port | int) ))))) -}} +{{- range $externalName, $external := $values.listeners.http.external -}} +{{- if (get (fromJson (include "redpanda.HTTPExternal.IsEnabled" (dict "a" (list $external) ))) "r") -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" (printf "http-%.8s" (lower $externalName)) "containerPort" ($external.port | int) ))))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "kafka" "containerPort" ($values.listeners.kafka.port | int) ))))) -}} +{{- range $externalName, $external := $values.listeners.kafka.external -}} +{{- if (get (fromJson (include "redpanda.KafkaExternal.IsEnabled" (dict "a" (list $external) ))) "r") -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" (printf "kafka-%.8s" (lower $externalName)) "containerPort" ($external.port | int) ))))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "rpc" "containerPort" ($values.listeners.rpc.port | int) ))))) -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" "schemaregistry" "containerPort" ($values.listeners.schemaRegistry.port | int) ))))) -}} +{{- range $externalName, $external := $values.listeners.schemaRegistry.external -}} +{{- if (get (fromJson (include "redpanda.SchemaRegistryExternal.IsEnabled" (dict "a" (list $external) ))) "r") -}} +{{- $_ := (set $container "ports" (concat (default (list ) $container.ports) (list (mustMergeOverwrite (dict "containerPort" 0 ) (dict "name" (printf "schema-%.8s" (lower $externalName)) "containerPort" ($external.port | int) ))))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (and (get (fromJson (include "redpanda.Storage.IsTieredStorageEnabled" (dict "a" (list $values.storage) ))) "r") (ne (get (fromJson (include "redpanda.Storage.TieredMountType" (dict "a" (list $values.storage) ))) "r") "none")) -}} +{{- $name := "tiered-storage-dir" -}} +{{- if (and (ne (toJson $values.storage.persistentVolume) "null") (ne $values.storage.persistentVolume.nameOverwrite "")) -}} +{{- $name = $values.storage.persistentVolume.nameOverwrite -}} +{{- end -}} +{{- $_ := (set $container "volumeMounts" (concat (default (list ) $container.volumeMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" $name "mountPath" (get (fromJson (include "redpanda.Storage.TieredCacheDirectory" (dict "a" (list $values.storage $dot) ))) "r") ))))) -}} +{{- end -}} +{{- $_ := (set $container.resources "limits" (dict "cpu" $values.resources.cpu.cores "memory" $values.resources.memory.container.max )) -}} +{{- if (ne (toJson $values.resources.memory.container.min) "null") -}} +{{- $_ := (set $container.resources "requests" (dict "cpu" $values.resources.cpu.cores "memory" $values.resources.memory.container.min )) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $container) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.adminApiURLs" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf `${SERVICE_NAME}.%s:%d` (get (fromJson (include "redpanda.InternalDomain" (dict "a" (list $dot) ))) "r") ($values.listeners.admin.port | int))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetContainerConfigWatcher" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.statefulset.sideCars.configWatcher.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "config-watcher" "image" (printf `%s:%s` $values.image.repository (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r")) "command" (list `/bin/sh`) "args" (list `-c` `trap "exit 0" TERM; exec /etc/secrets/config-watcher/scripts/sasl-user.sh & wait $!`) "env" (get (fromJson (include "redpanda.rpkEnvVars" (dict "a" (list $dot (coalesce nil)) ))) "r") "resources" $values.statefulset.sideCars.configWatcher.resources "securityContext" $values.statefulset.sideCars.configWatcher.securityContext "volumeMounts" (concat (default (list ) (concat (default (list ) (get (fromJson (include "redpanda.CommonMounts" (dict "a" (list $dot) ))) "r")) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" "config" "mountPath" "/etc/redpanda" )) (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" (printf `%s-config-watcher` (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) "mountPath" "/etc/secrets/config-watcher/scripts" ))))) (default (list ) (get (fromJson (include "redpanda.templateToVolumeMounts" (dict "a" (list $dot $values.statefulset.sideCars.configWatcher.extraVolumeMounts) ))) "r"))) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetContainerControllers" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not $values.rbac.enabled) (not $values.statefulset.sideCars.controllers.enabled)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $volumeMounts := (list ) -}} +{{- if (and (and (and $values.rbac.enabled $values.statefulset.sideCars.controllers.enabled) $values.statefulset.sideCars.controllers.createRBAC) (not (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.serviceAccount.automountServiceAccountToken false) ))) "r"))) -}} +{{- $mountName := "kube-api-access" -}} +{{- range $_, $vol := (get (fromJson (include "redpanda.StatefulSetVolumes" (dict "a" (list $dot) ))) "r") -}} +{{- if (hasPrefix $vol.name (printf "%s%s" "kube-api-access" "-")) -}} +{{- $mountName = $vol.name -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $volumeMounts = (concat (default (list ) $volumeMounts) (list (mustMergeOverwrite (dict "name" "" "mountPath" "" ) (dict "name" $mountName "readOnly" true "mountPath" "/var/run/secrets/kubernetes.io/serviceaccount" )))) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" "resources" (dict ) ) (dict "name" "redpanda-controllers" "image" (printf `%s:%s` $values.statefulset.sideCars.controllers.image.repository $values.statefulset.sideCars.controllers.image.tag) "command" (list `/manager`) "args" (list `--operator-mode=false` (printf `--namespace=%s` $dot.Release.Namespace) (printf `--health-probe-bind-address=%s` $values.statefulset.sideCars.controllers.healthProbeAddress) (printf `--metrics-bind-address=%s` $values.statefulset.sideCars.controllers.metricsAddress) (printf `--additional-controllers=%s` (join "," $values.statefulset.sideCars.controllers.run))) "env" (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_HELM_RELEASE_NAME" "value" $dot.Release.Name ))) "resources" $values.statefulset.sideCars.controllers.resources "securityContext" $values.statefulset.sideCars.controllers.securityContext "volumeMounts" $volumeMounts ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.rpkEnvVars" -}} +{{- $dot := (index .a 0) -}} +{{- $envVars := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (and (ne (toJson $values.auth.sasl) "null") $values.auth.sasl.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $envVars) (default (list ) (get (fromJson (include "redpanda.BootstrapUser.RpkEnvironment" (dict "a" (list $values.auth.sasl.bootstrapUser (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) ))) "r")))) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $envVars) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.bootstrapEnvVars" -}} +{{- $dot := (index .a 0) -}} +{{- $envVars := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (and (ne (toJson $values.auth.sasl) "null") $values.auth.sasl.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) $envVars) (default (list ) (get (fromJson (include "redpanda.BootstrapUser.BootstrapEnvironment" (dict "a" (list $values.auth.sasl.bootstrapUser (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r")) ))) "r")))) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $envVars) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.templateToVolumeMounts" -}} +{{- $dot := (index .a 0) -}} +{{- $template := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (tpl $template $dot) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (fromYamlArray $result)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.templateToVolumes" -}} +{{- $dot := (index .a 0) -}} +{{- $template := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (tpl $template $dot) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (fromYamlArray $result)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.templateToContainers" -}} +{{- $dot := (index .a 0) -}} +{{- $template := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (tpl $template $dot) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (fromYamlArray $result)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StatefulSet" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (and (not (get (fromJson (include "redpanda.RedpandaAtLeast_22_2_0" (dict "a" (list $dot) ))) "r")) (not $values.force)) -}} +{{- $sv := (get (fromJson (include "redpanda.semver" (dict "a" (list $dot) ))) "r") -}} +{{- $_ := (fail (printf "Error: The Redpanda version (%s) is no longer supported \nTo accept this risk, run the upgrade again adding `--force=true`\n" $sv)) -}} +{{- end -}} +{{- $ss := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "serviceName" "" "updateStrategy" (dict ) ) "status" (dict "replicas" 0 "availableReplicas" 0 ) ) (mustMergeOverwrite (dict ) (dict "apiVersion" "apps/v1" "kind" "StatefulSet" )) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (get (fromJson (include "redpanda.Fullname" (dict "a" (list $dot) ))) "r") "namespace" $dot.Release.Namespace "labels" (get (fromJson (include "redpanda.FullLabels" (dict "a" (list $dot) ))) "r") )) "spec" (mustMergeOverwrite (dict "selector" (coalesce nil) "template" (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) "serviceName" "" "updateStrategy" (dict ) ) (dict "selector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") )) "serviceName" (get (fromJson (include "redpanda.ServiceName" (dict "a" (list $dot) ))) "r") "replicas" ($values.statefulset.replicas | int) "updateStrategy" $values.statefulset.updateStrategy "podManagementPolicy" "Parallel" "template" (get (fromJson (include "redpanda.StrategicMergePatch" (dict "a" (list $values.statefulset.podTemplate (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "containers" (coalesce nil) ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "labels" (get (fromJson (include "redpanda.StatefulSetPodLabels" (dict "a" (list $dot) ))) "r") "annotations" (get (fromJson (include "redpanda.StatefulSetPodAnnotations" (dict "a" (list $dot (get (fromJson (include "redpanda.statefulSetChecksumAnnotation" (dict "a" (list $dot) ))) "r")) ))) "r") )) "spec" (mustMergeOverwrite (dict "containers" (coalesce nil) ) (dict "automountServiceAccountToken" false "terminationGracePeriodSeconds" ($values.statefulset.terminationGracePeriodSeconds | int64) "securityContext" (get (fromJson (include "redpanda.PodSecurityContext" (dict "a" (list $dot) ))) "r") "serviceAccountName" (get (fromJson (include "redpanda.ServiceAccountName" (dict "a" (list $dot) ))) "r") "imagePullSecrets" (default (coalesce nil) $values.imagePullSecrets) "initContainers" (get (fromJson (include "redpanda.StatefulSetInitContainers" (dict "a" (list $dot) ))) "r") "containers" (get (fromJson (include "redpanda.StatefulSetContainers" (dict "a" (list $dot) ))) "r") "volumes" (get (fromJson (include "redpanda.StatefulSetVolumes" (dict "a" (list $dot) ))) "r") "topologySpreadConstraints" (get (fromJson (include "redpanda.statefulSetTopologySpreadConstraints" (dict "a" (list $dot) ))) "r") "nodeSelector" (get (fromJson (include "redpanda.statefulSetNodeSelectors" (dict "a" (list $dot) ))) "r") "affinity" (get (fromJson (include "redpanda.statefulSetAffinity" (dict "a" (list $dot) ))) "r") "priorityClassName" $values.statefulset.priorityClassName "tolerations" (get (fromJson (include "redpanda.statefulSetTolerations" (dict "a" (list $dot) ))) "r") )) ))) ))) "r") "volumeClaimTemplates" (coalesce nil) )) )) -}} +{{- if (or $values.storage.persistentVolume.enabled ((and (get (fromJson (include "redpanda.Storage.IsTieredStorageEnabled" (dict "a" (list $values.storage) ))) "r") (eq (get (fromJson (include "redpanda.Storage.TieredMountType" (dict "a" (list $values.storage) ))) "r") "persistentVolume")))) -}} +{{- $t_13 := (get (fromJson (include "redpanda.volumeClaimTemplateDatadir" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $t_13) "null") -}} +{{- $_ := (set $ss.spec "volumeClaimTemplates" (concat (default (list ) $ss.spec.volumeClaimTemplates) (list $t_13))) -}} +{{- end -}} +{{- $t_14 := (get (fromJson (include "redpanda.volumeClaimTemplateTieredStorageDir" (dict "a" (list $dot) ))) "r") -}} +{{- if (ne (toJson $t_14) "null") -}} +{{- $_ := (set $ss.spec "volumeClaimTemplates" (concat (default (list ) $ss.spec.volumeClaimTemplates) (list $t_14))) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $ss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.semver" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (trimPrefix "v" (get (fromJson (include "redpanda.Tag" (dict "a" (list $dot) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetChecksumAnnotation" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $dependencies := (coalesce nil) -}} +{{- $dependencies = (concat (default (list ) $dependencies) (list (get (fromJson (include "redpanda.RedpandaConfigFile" (dict "a" (list $dot false) ))) "r"))) -}} +{{- if $values.external.enabled -}} +{{- $dependencies = (concat (default (list ) $dependencies) (list (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $values.external.domain "") ))) "r"))) -}} +{{- if (empty $values.external.addresses) -}} +{{- $dependencies = (concat (default (list ) $dependencies) (list "")) -}} +{{- else -}} +{{- $dependencies = (concat (default (list ) $dependencies) (list $values.external.addresses)) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (sha256sum (toJson $dependencies))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetTolerations" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default $values.tolerations $values.statefulset.tolerations)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetNodeSelectors" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (default $values.statefulset.nodeSelector $values.nodeSelector)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetAffinity" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $affinity := (mustMergeOverwrite (dict ) (dict )) -}} +{{- if (not (empty $values.statefulset.nodeAffinity)) -}} +{{- $_ := (set $affinity "nodeAffinity" $values.statefulset.nodeAffinity) -}} +{{- else -}}{{- if (not (empty $values.affinity.nodeAffinity)) -}} +{{- $_ := (set $affinity "nodeAffinity" $values.affinity.nodeAffinity) -}} +{{- end -}} +{{- end -}} +{{- if (not (empty $values.statefulset.podAffinity)) -}} +{{- $_ := (set $affinity "podAffinity" $values.statefulset.podAffinity) -}} +{{- else -}}{{- if (not (empty $values.affinity.podAffinity)) -}} +{{- $_ := (set $affinity "podAffinity" $values.affinity.podAffinity) -}} +{{- end -}} +{{- end -}} +{{- if (not (empty $values.statefulset.podAntiAffinity)) -}} +{{- $_ := (set $affinity "podAntiAffinity" (mustMergeOverwrite (dict ) (dict ))) -}} +{{- if (eq $values.statefulset.podAntiAffinity.type "hard") -}} +{{- $_ := (set $affinity.podAntiAffinity "requiredDuringSchedulingIgnoredDuringExecution" (list (mustMergeOverwrite (dict "topologyKey" "" ) (dict "topologyKey" $values.statefulset.podAntiAffinity.topologyKey "labelSelector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") )) )))) -}} +{{- else -}}{{- if (eq $values.statefulset.podAntiAffinity.type "soft") -}} +{{- $_ := (set $affinity.podAntiAffinity "preferredDuringSchedulingIgnoredDuringExecution" (list (mustMergeOverwrite (dict "weight" 0 "podAffinityTerm" (dict "topologyKey" "" ) ) (dict "weight" ($values.statefulset.podAntiAffinity.weight | int) "podAffinityTerm" (mustMergeOverwrite (dict "topologyKey" "" ) (dict "topologyKey" $values.statefulset.podAntiAffinity.topologyKey "labelSelector" (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") )) )) )))) -}} +{{- else -}}{{- if (eq $values.statefulset.podAntiAffinity.type "custom") -}} +{{- $_ := (set $affinity "podAntiAffinity" $values.statefulset.podAntiAffinity.custom) -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- else -}}{{- if (not (empty $values.affinity.podAntiAffinity)) -}} +{{- $_ := (set $affinity "podAntiAffinity" $values.affinity.podAntiAffinity) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $affinity) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.volumeClaimTemplateDatadir" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (not $values.storage.persistentVolume.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $pvc := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "resources" (dict ) ) "status" (dict ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" "datadir" "labels" (merge (dict ) (dict `app.kubernetes.io/name` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") `app.kubernetes.io/instance` $dot.Release.Name `app.kubernetes.io/component` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") ) $values.storage.persistentVolume.labels $values.commonLabels) "annotations" (default (coalesce nil) $values.storage.persistentVolume.annotations) )) "spec" (mustMergeOverwrite (dict "resources" (dict ) ) (dict "accessModes" (list "ReadWriteOnce") "resources" (mustMergeOverwrite (dict ) (dict "requests" (dict "storage" $values.storage.persistentVolume.size ) )) )) )) -}} +{{- if (not (empty $values.storage.persistentVolume.storageClass)) -}} +{{- if (eq $values.storage.persistentVolume.storageClass "-") -}} +{{- $_ := (set $pvc.spec "storageClassName" "") -}} +{{- else -}} +{{- $_ := (set $pvc.spec "storageClassName" $values.storage.persistentVolume.storageClass) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $pvc) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.volumeClaimTemplateTieredStorageDir" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- if (or (not (get (fromJson (include "redpanda.Storage.IsTieredStorageEnabled" (dict "a" (list $values.storage) ))) "r")) (ne (get (fromJson (include "redpanda.Storage.TieredMountType" (dict "a" (list $values.storage) ))) "r") "persistentVolume")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $pvc := (mustMergeOverwrite (dict "metadata" (dict "creationTimestamp" (coalesce nil) ) "spec" (dict "resources" (dict ) ) "status" (dict ) ) (dict "metadata" (mustMergeOverwrite (dict "creationTimestamp" (coalesce nil) ) (dict "name" (default "tiered-storage-dir" $values.storage.persistentVolume.nameOverwrite) "labels" (merge (dict ) (dict `app.kubernetes.io/name` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") `app.kubernetes.io/instance` $dot.Release.Name `app.kubernetes.io/component` (get (fromJson (include "redpanda.Name" (dict "a" (list $dot) ))) "r") ) (get (fromJson (include "redpanda.Storage.TieredPersistentVolumeLabels" (dict "a" (list $values.storage) ))) "r") $values.commonLabels) "annotations" (default (coalesce nil) (get (fromJson (include "redpanda.Storage.TieredPersistentVolumeAnnotations" (dict "a" (list $values.storage) ))) "r")) )) "spec" (mustMergeOverwrite (dict "resources" (dict ) ) (dict "accessModes" (list "ReadWriteOnce") "resources" (mustMergeOverwrite (dict ) (dict "requests" (dict "storage" (index (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r") `cloud_storage_cache_size`) ) )) )) )) -}} +{{- $sc_15 := (get (fromJson (include "redpanda.Storage.TieredPersistentVolumeStorageClass" (dict "a" (list $values.storage) ))) "r") -}} +{{- if (eq $sc_15 "-") -}} +{{- $_ := (set $pvc.spec "storageClassName" "") -}} +{{- else -}}{{- if (not (empty $sc_15)) -}} +{{- $_ := (set $pvc.spec "storageClassName" $sc_15) -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $pvc) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.statefulSetTopologySpreadConstraints" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $result := (coalesce nil) -}} +{{- $labelSelector := (mustMergeOverwrite (dict ) (dict "matchLabels" (get (fromJson (include "redpanda.StatefulSetPodLabelsSelector" (dict "a" (list $dot) ))) "r") )) -}} +{{- range $_, $v := $values.statefulset.topologySpreadConstraints -}} +{{- $result = (concat (default (list ) $result) (list (mustMergeOverwrite (dict "maxSkew" 0 "topologyKey" "" "whenUnsatisfiable" "" ) (dict "maxSkew" ($v.maxSkew | int) "topologyKey" $v.topologyKey "whenUnsatisfiable" $v.whenUnsatisfiable "labelSelector" $labelSelector )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.StorageTieredConfig" -}} +{{- $dot := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/_values.go.tpl b/charts/redpanda/redpanda/5.9.10/templates/_values.go.tpl new file mode 100644 index 0000000000..703f35516a --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/_values.go.tpl @@ -0,0 +1,1351 @@ +{{- /* Generated from "values.go" */ -}} + +{{- define "redpanda.AuditLogging.Translate" -}} +{{- $a := (index .a 0) -}} +{{- $dot := (index .a 1) -}} +{{- $isSASLEnabled := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- if (not (get (fromJson (include "redpanda.RedpandaAtLeast_23_3_0" (dict "a" (list $dot) ))) "r")) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- $enabled := (and $a.enabled $isSASLEnabled) -}} +{{- $_ := (set $result "audit_enabled" $enabled) -}} +{{- if (not $enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (ne (($a.clientMaxBufferSize | int) | int) (16777216 | int)) -}} +{{- $_ := (set $result "audit_client_max_buffer_size" ($a.clientMaxBufferSize | int)) -}} +{{- end -}} +{{- if (ne (($a.queueDrainIntervalMs | int) | int) (500 | int)) -}} +{{- $_ := (set $result "audit_queue_drain_interval_ms" ($a.queueDrainIntervalMs | int)) -}} +{{- end -}} +{{- if (ne (($a.queueMaxBufferSizePerShard | int) | int) (1048576 | int)) -}} +{{- $_ := (set $result "audit_queue_max_buffer_size_per_shard" ($a.queueMaxBufferSizePerShard | int)) -}} +{{- end -}} +{{- if (ne (($a.partitions | int) | int) (12 | int)) -}} +{{- $_ := (set $result "audit_log_num_partitions" ($a.partitions | int)) -}} +{{- end -}} +{{- if (ne ($a.replicationFactor | int) (0 | int)) -}} +{{- $_ := (set $result "audit_log_replication_factor" ($a.replicationFactor | int)) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $a.enabledEventTypes) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $result "audit_enabled_event_types" $a.enabledEventTypes) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $a.excludedTopics) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $result "audit_excluded_topics" $a.excludedTopics) -}} +{{- end -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $a.excludedPrincipals) ))) "r") | int) (0 | int)) -}} +{{- $_ := (set $result "audit_excluded_principals" $a.excludedPrincipals) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Auth.IsSASLEnabled" -}} +{{- $a := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $a.sasl) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" false) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $a.sasl.enabled) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Auth.Translate" -}} +{{- $a := (index .a 0) -}} +{{- $isSASLEnabled := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (not $isSASLEnabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $users := (list (get (fromJson (include "redpanda.BootstrapUser.Username" (dict "a" (list $a.sasl.bootstrapUser) ))) "r")) -}} +{{- range $_, $u := $a.sasl.users -}} +{{- $users = (concat (default (list ) $users) (list $u.name)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict "superusers" $users )) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Logging.Translate" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- $clusterID_1 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.usageStats.clusterId "") ))) "r") -}} +{{- if (ne $clusterID_1 "") -}} +{{- $_ := (set $result "cluster_id" $clusterID_1) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.RedpandaResources.GetOverProvisionValue" -}} +{{- $rr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (lt ((get (fromJson (include "_shims.resource_MilliValue" (dict "a" (list $rr.cpu.cores) ))) "r") | int64) (1000 | int64)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" true) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $rr.cpu.overprovisioned false) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.IsTieredStorageEnabled" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $conf := (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $s) ))) "r") -}} +{{- $tmp_tuple_3 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $conf "cloud_storage_enabled" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_3.T2 -}} +{{- $b := $tmp_tuple_3.T1 -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and $ok (get (fromJson (include "_shims.typeassertion" (dict "a" (list "bool" $b) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.GetTieredStorageConfig" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $s.tieredConfig) ))) "r") | int) (0 | int)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tieredConfig) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tiered.config) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.GetTieredStorageHostPath" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $hp := $s.tieredStorageHostPath -}} +{{- if (empty $hp) -}} +{{- $hp = $s.tiered.hostPath -}} +{{- end -}} +{{- if (empty $hp) -}} +{{- $_ := (fail (printf `storage.tiered.mountType is "%s" but storage.tiered.hostPath is empty` $s.tiered.mountType)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $hp) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.TieredCacheDirectory" -}} +{{- $s := (index .a 0) -}} +{{- $dot := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $values := $dot.Values.AsMap -}} +{{- $tmp_tuple_4 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "string" (index $values.config.node "cloud_storage_cache_directory") "") ))) "r")) ))) "r") -}} +{{- $ok_3 := $tmp_tuple_4.T2 -}} +{{- $dir_2 := $tmp_tuple_4.T1 -}} +{{- if $ok_3 -}} +{{- $_is_returning = true -}} +{{- (dict "r" $dir_2) | toJson -}} +{{- break -}} +{{- end -}} +{{- $tieredConfig := (get (fromJson (include "redpanda.Storage.GetTieredStorageConfig" (dict "a" (list $values.storage) ))) "r") -}} +{{- $tmp_tuple_5 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "string" (index $tieredConfig "cloud_storage_cache_directory") "") ))) "r")) ))) "r") -}} +{{- $ok_5 := $tmp_tuple_5.T2 -}} +{{- $dir_4 := $tmp_tuple_5.T1 -}} +{{- if $ok_5 -}} +{{- $_is_returning = true -}} +{{- (dict "r" $dir_4) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "/var/lib/redpanda/data/cloud_storage_cache") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.TieredMountType" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (and (ne (toJson $s.tieredStoragePersistentVolume) "null") $s.tieredStoragePersistentVolume.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" "persistentVolume") | toJson -}} +{{- break -}} +{{- end -}} +{{- if (not (empty $s.tieredStorageHostPath)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" "hostPath") | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tiered.mountType) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.TieredPersistentVolumeLabels" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $s.tieredStoragePersistentVolume) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tieredStoragePersistentVolume.labels) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tiered.persistentVolume.labels) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.TieredPersistentVolumeAnnotations" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $s.tieredStoragePersistentVolume) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tieredStoragePersistentVolume.annotations) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tiered.persistentVolume.annotations) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.TieredPersistentVolumeStorageClass" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $s.tieredStoragePersistentVolume) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tieredStoragePersistentVolume.storageClass) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $s.tiered.persistentVolume.storageClass) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Storage.StorageMinFreeBytes" -}} +{{- $s := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (and (ne (toJson $s.persistentVolume) "null") (not $s.persistentVolume.enabled)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (5368709120 | int)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $minimumFreeBytes := ((mulf (((get (fromJson (include "_shims.resource_Value" (dict "a" (list $s.persistentVolume.size) ))) "r") | int64) | float64) 0.05) | float64) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (min (5368709120 | int) ($minimumFreeBytes | int64))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Tuning.Translate" -}} +{{- $t := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- $s := (toJson $t) -}} +{{- $tune := (fromJson $s) -}} +{{- $tmp_tuple_7 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list (printf "map[%s]%s" "string" "interface {}") $tune (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_7.T2 -}} +{{- $m := $tmp_tuple_7.T1 -}} +{{- if (not $ok) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (dict )) | toJson -}} +{{- break -}} +{{- end -}} +{{- range $k, $v := $m -}} +{{- $_ := (set $result $k $v) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Listeners.CreateSeedServers" -}} +{{- $l := (index .a 0) -}} +{{- $replicas := (index .a 1) -}} +{{- $fullname := (index .a 2) -}} +{{- $internalDomain := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (coalesce nil) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) ($replicas|int) (1|int) -}} +{{- $result = (concat (default (list ) $result) (list (dict "host" (dict "address" (printf "%s-%d.%s" $fullname $i $internalDomain) "port" ($l.rpc.port | int) ) ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Listeners.AdminList" -}} +{{- $l := (index .a 0) -}} +{{- $replicas := (index .a 1) -}} +{{- $fullname := (index .a 2) -}} +{{- $internalDomain := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.ServerList" (dict "a" (list $replicas "" $fullname $internalDomain ($l.admin.port | int)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Listeners.SchemaRegistryList" -}} +{{- $l := (index .a 0) -}} +{{- $replicas := (index .a 1) -}} +{{- $fullname := (index .a 2) -}} +{{- $internalDomain := (index .a 3) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.ServerList" (dict "a" (list $replicas "" $fullname $internalDomain ($l.schemaRegistry.port | int)) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ServerList" -}} +{{- $replicas := (index .a 0) -}} +{{- $prefix := (index .a 1) -}} +{{- $fullname := (index .a 2) -}} +{{- $internalDomain := (index .a 3) -}} +{{- $port := (index .a 4) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (coalesce nil) -}} +{{- range $_, $i := untilStep (((0 | int) | int)|int) ($replicas|int) (1|int) -}} +{{- $result = (concat (default (list ) $result) (list (printf "%s%s-%d.%s:%d" $prefix $fullname $i $internalDomain ($port | int)))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Listeners.TrustStoreVolume" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $cmSources := (dict ) -}} +{{- $secretSources := (dict ) -}} +{{- range $_, $ts := (get (fromJson (include "redpanda.Listeners.TrustStores" (dict "a" (list $l $tls) ))) "r") -}} +{{- $projection := (get (fromJson (include "redpanda.TrustStore.VolumeProjection" (dict "a" (list $ts) ))) "r") -}} +{{- if (ne (toJson $projection.secret) "null") -}} +{{- $_ := (set $secretSources $projection.secret.name (concat (default (list ) (index $secretSources $projection.secret.name)) (default (list ) $projection.secret.items))) -}} +{{- else -}} +{{- $_ := (set $cmSources $projection.configMap.name (concat (default (list ) (index $cmSources $projection.configMap.name)) (default (list ) $projection.configMap.items))) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $sources := (coalesce nil) -}} +{{- range $_, $name := (sortAlpha (keys $cmSources)) -}} +{{- $keys := (index $cmSources $name) -}} +{{- $sources = (concat (default (list ) $sources) (list (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" $name )) (dict "items" (get (fromJson (include "redpanda.dedupKeyToPaths" (dict "a" (list $keys) ))) "r") )) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- range $_, $name := (sortAlpha (keys $secretSources)) -}} +{{- $keys := (index $secretSources $name) -}} +{{- $sources = (concat (default (list ) $sources) (list (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" $name )) (dict "items" (get (fromJson (include "redpanda.dedupKeyToPaths" (dict "a" (list $keys) ))) "r") )) )))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- if (lt ((get (fromJson (include "_shims.len" (dict "a" (list $sources) ))) "r") | int) (1 | int)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "name" "" ) (mustMergeOverwrite (dict ) (dict "projected" (mustMergeOverwrite (dict "sources" (coalesce nil) ) (dict "sources" $sources )) )) (dict "name" "truststores" ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.dedupKeyToPaths" -}} +{{- $items := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $seen := (dict ) -}} +{{- $deduped := (coalesce nil) -}} +{{- range $_, $item := $items -}} +{{- $tmp_tuple_8 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $seen $item.key (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok_6 := $tmp_tuple_8.T2 -}} +{{- if $ok_6 -}} +{{- continue -}} +{{- end -}} +{{- $deduped = (concat (default (list ) $deduped) (list $item)) -}} +{{- $_ := (set $seen $item.key true) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $deduped) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Listeners.TrustStores" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tss := (get (fromJson (include "redpanda.KafkaListeners.TrustStores" (dict "a" (list $l.kafka $tls) ))) "r") -}} +{{- $tss = (concat (default (list ) $tss) (default (list ) (get (fromJson (include "redpanda.AdminListeners.TrustStores" (dict "a" (list $l.admin $tls) ))) "r"))) -}} +{{- $tss = (concat (default (list ) $tss) (default (list ) (get (fromJson (include "redpanda.HTTPListeners.TrustStores" (dict "a" (list $l.http $tls) ))) "r"))) -}} +{{- $tss = (concat (default (list ) $tss) (default (list ) (get (fromJson (include "redpanda.SchemaRegistryListeners.TrustStores" (dict "a" (list $l.schemaRegistry $tls) ))) "r"))) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.Config.CreateRPKConfiguration" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- range $k, $v := $c.rpk -}} +{{- $_ := (set $result $k $v) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TLSCertMap.MustGet" -}} +{{- $m := (index .a 0) -}} +{{- $name := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_9 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $m $name (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_9.T2 -}} +{{- $cert := $tmp_tuple_9.T1 -}} +{{- if (not $ok) -}} +{{- $_ := (fail (printf "Certificate %q referenced, but not found in the tls.certs map" $name)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $cert) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapUser.BootstrapEnvironment" -}} +{{- $b := (index .a 0) -}} +{{- $fullname := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (concat (default (list ) (get (fromJson (include "redpanda.BootstrapUser.RpkEnvironment" (dict "a" (list $b $fullname) ))) "r")) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "RP_BOOTSTRAP_USER" "value" "$(RPK_USER):$(RPK_PASS):$(RPK_SASL_MECHANISM)" ))))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapUser.Username" -}} +{{- $b := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $b.name) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $b.name) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "kubernetes-controller") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapUser.RpkEnvironment" -}} +{{- $b := (index .a 0) -}} +{{- $fullname := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "RPK_PASS" "valueFrom" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (get (fromJson (include "redpanda.BootstrapUser.SecretKeySelector" (dict "a" (list $b $fullname) ))) "r") )) )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "RPK_USER" "value" (get (fromJson (include "redpanda.BootstrapUser.Username" (dict "a" (list $b) ))) "r") )) (mustMergeOverwrite (dict "name" "" ) (dict "name" "RPK_SASL_MECHANISM" "value" (get (fromJson (include "redpanda.BootstrapUser.GetMechanism" (dict "a" (list $b) ))) "r") )))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapUser.GetMechanism" -}} +{{- $b := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq $b.mechanism "") -}} +{{- $_is_returning = true -}} +{{- (dict "r" "SCRAM-SHA-256") | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $b.mechanism) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.BootstrapUser.SecretKeySelector" -}} +{{- $b := (index .a 0) -}} +{{- $fullname := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $b.secretKeyRef) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" $b.secretKeyRef) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" (printf "%s-bootstrap-user" $fullname) )) (dict "key" "password" ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TrustStore.TrustStoreFilePath" -}} +{{- $t := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s" "/etc/truststores" (get (fromJson (include "redpanda.TrustStore.RelativePath" (dict "a" (list $t) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TrustStore.RelativePath" -}} +{{- $t := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $t.configMapKeyRef) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "configmaps/%s-%s" $t.configMapKeyRef.name $t.configMapKeyRef.key)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "secrets/%s-%s" $t.secretKeyRef.name $t.secretKeyRef.key)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TrustStore.VolumeProjection" -}} +{{- $t := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $t.configMapKeyRef) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict ) (dict "configMap" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" $t.configMapKeyRef.name )) (dict "items" (list (mustMergeOverwrite (dict "key" "" "path" "" ) (dict "key" $t.configMapKeyRef.key "path" (get (fromJson (include "redpanda.TrustStore.RelativePath" (dict "a" (list $t) ))) "r") ))) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict ) (dict "secret" (mustMergeOverwrite (dict ) (mustMergeOverwrite (dict ) (dict "name" $t.secretKeyRef.name )) (dict "items" (list (mustMergeOverwrite (dict "key" "" "path" "" ) (dict "key" $t.secretKeyRef.key "path" (get (fromJson (include "redpanda.TrustStore.RelativePath" (dict "a" (list $t) ))) "r") ))) )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.InternalTLS.IsEnabled" -}} +{{- $t := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $t.enabled $tls.enabled) ))) "r") (ne $t.cert ""))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.InternalTLS.TrustStoreFilePath" -}} +{{- $t := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $t.trustStore) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.TrustStore.TrustStoreFilePath" (dict "a" (list $t.trustStore) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) $t.cert) ))) "r").caEnabled -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s/ca.crt" "/etc/tls/certs" $t.cert)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "/etc/ssl/certs/ca-certificates.crt") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.InternalTLS.ServerCAPath" -}} +{{- $t := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) $t.cert) ))) "r").caEnabled -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s/ca.crt" "/etc/tls/certs" $t.cert)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s/tls.crt" "/etc/tls/certs" $t.cert)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ExternalTLS.GetCert" -}} +{{- $t := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- $tls := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $t $i) ))) "r")) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ExternalTLS.GetCertName" -}} +{{- $t := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $t.cert $i.cert) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ExternalTLS.TrustStoreFilePath" -}} +{{- $t := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- $tls := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (ne (toJson $t.trustStore) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (get (fromJson (include "redpanda.TrustStore.TrustStoreFilePath" (dict "a" (list $t.trustStore) ))) "r")) | toJson -}} +{{- break -}} +{{- end -}} +{{- if (get (fromJson (include "redpanda.ExternalTLS.GetCert" (dict "a" (list $t $i $tls) ))) "r").caEnabled -}} +{{- $_is_returning = true -}} +{{- (dict "r" (printf "%s/%s/ca.crt" "/etc/tls/certs" (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $t $i) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" "/etc/ssl/certs/ca-certificates.crt") | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ExternalTLS.IsEnabled" -}} +{{- $t := (index .a 0) -}} +{{- $i := (index .a 1) -}} +{{- $tls := (index .a 2) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $t) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" false) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (ne (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $t $i) ))) "r") "") (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $t.enabled (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $i $tls) ))) "r")) ))) "r"))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.AdminListeners.ConsoleTLS" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $t := (mustMergeOverwrite (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false ) (dict "enabled" (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $l.tls $tls) ))) "r") )) -}} +{{- if (not $t.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $adminAPIPrefix := (printf "%s/%s" "/etc/tls/certs" $l.tls.cert) -}} +{{- if (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) $l.tls.cert) ))) "r").caEnabled -}} +{{- $_ := (set $t "caFilepath" (printf "%s/ca.crt" $adminAPIPrefix)) -}} +{{- else -}} +{{- $_ := (set $t "caFilepath" (printf "%s/tls.crt" $adminAPIPrefix)) -}} +{{- end -}} +{{- if (not $l.tls.requireClientAuth) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $t "certFilepath" (printf "%s/tls.crt" $adminAPIPrefix)) -}} +{{- $_ := (set $t "keyFilepath" (printf "%s/tls.key" $adminAPIPrefix)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.AdminListeners.Listeners" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $admin := (list (get (fromJson (include "redpanda.createInternalListenerCfg" (dict "a" (list ($l.port | int)) ))) "r")) -}} +{{- range $k, $lis := $l.external -}} +{{- if (not (get (fromJson (include "redpanda.AdminExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $admin = (concat (default (list ) $admin) (list (dict "name" $k "port" ($lis.port | int) "address" "0.0.0.0" ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $admin) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.AdminListeners.ListenersTLS" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $admin := (list ) -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerTLSCfg" (dict "a" (list $tls $l.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $internal) ))) "r") | int) (0 | int)) -}} +{{- $admin = (concat (default (list ) $admin) (list $internal)) -}} +{{- end -}} +{{- range $k, $lis := $l.external -}} +{{- if (or (not (get (fromJson (include "redpanda.AdminExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- $certName := (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $lis.tls $l.tls) ))) "r") -}} +{{- $admin = (concat (default (list ) $admin) (list (dict "name" $k "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $certName) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $certName) "require_client_auth" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $lis.tls.requireClientAuth false) ))) "r") "truststore_file" (get (fromJson (include "redpanda.ExternalTLS.TrustStoreFilePath" (dict "a" (list $lis.tls $l.tls $tls) ))) "r") ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $admin) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.AdminListeners.TrustStores" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tss := (list ) -}} +{{- if (and (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $l.tls $tls) ))) "r") (ne (toJson $l.tls.trustStore) "null")) -}} +{{- $tss = (concat (default (list ) $tss) (list $l.tls.trustStore)) -}} +{{- end -}} +{{- range $_, $key := (sortAlpha (keys $l.external)) -}} +{{- $lis := (index $l.external $key) -}} +{{- if (or (or (not (get (fromJson (include "redpanda.AdminExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) (eq (toJson $lis.tls.trustStore) "null")) -}} +{{- continue -}} +{{- end -}} +{{- $tss = (concat (default (list ) $tss) (list $lis.tls.trustStore)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.AdminExternal.IsEnabled" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.enabled true) ))) "r") (gt ($l.port | int) (0 | int)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.HTTPListeners.Listeners" -}} +{{- $l := (index .a 0) -}} +{{- $saslEnabled := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerCfg" (dict "a" (list ($l.port | int)) ))) "r") -}} +{{- if $saslEnabled -}} +{{- $_ := (set $internal "authentication_method" "http_basic") -}} +{{- end -}} +{{- $am_7 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_7 "") -}} +{{- $_ := (set $internal "authentication_method" $am_7) -}} +{{- end -}} +{{- $result := (list $internal) -}} +{{- range $k, $l := $l.external -}} +{{- if (not (get (fromJson (include "redpanda.HTTPExternal.IsEnabled" (dict "a" (list $l) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $listener := (dict "name" $k "port" ($l.port | int) "address" "0.0.0.0" ) -}} +{{- if $saslEnabled -}} +{{- $_ := (set $listener "authentication_method" "http_basic") -}} +{{- end -}} +{{- $am_8 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_8 "") -}} +{{- $_ := (set $listener "authentication_method" $am_8) -}} +{{- end -}} +{{- $result = (concat (default (list ) $result) (list $listener)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.HTTPListeners.ListenersTLS" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $pp := (list ) -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerTLSCfg" (dict "a" (list $tls $l.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $internal) ))) "r") | int) (0 | int)) -}} +{{- $pp = (concat (default (list ) $pp) (list $internal)) -}} +{{- end -}} +{{- range $k, $lis := $l.external -}} +{{- if (or (not (get (fromJson (include "redpanda.HTTPExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- $certName := (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $lis.tls $l.tls) ))) "r") -}} +{{- $pp = (concat (default (list ) $pp) (list (dict "name" $k "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $certName) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $certName) "require_client_auth" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $lis.tls.requireClientAuth false) ))) "r") "truststore_file" (get (fromJson (include "redpanda.ExternalTLS.TrustStoreFilePath" (dict "a" (list $lis.tls $l.tls $tls) ))) "r") ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $pp) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.HTTPListeners.TrustStores" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tss := (coalesce nil) -}} +{{- if (and (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $l.tls $tls) ))) "r") (ne (toJson $l.tls.trustStore) "null")) -}} +{{- $tss = (concat (default (list ) $tss) (list $l.tls.trustStore)) -}} +{{- end -}} +{{- range $_, $key := (sortAlpha (keys $l.external)) -}} +{{- $lis := (index $l.external $key) -}} +{{- if (or (or (not (get (fromJson (include "redpanda.HTTPExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) (eq (toJson $lis.tls.trustStore) "null")) -}} +{{- continue -}} +{{- end -}} +{{- $tss = (concat (default (list ) $tss) (list $lis.tls.trustStore)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.HTTPExternal.IsEnabled" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.enabled true) ))) "r") (gt ($l.port | int) (0 | int)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.KafkaListeners.Listeners" -}} +{{- $l := (index .a 0) -}} +{{- $auth := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerCfg" (dict "a" (list ($l.port | int)) ))) "r") -}} +{{- if (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $auth) ))) "r") -}} +{{- $_ := (set $internal "authentication_method" "sasl") -}} +{{- end -}} +{{- $am_9 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_9 "") -}} +{{- $_ := (set $internal "authentication_method" $am_9) -}} +{{- end -}} +{{- $kafka := (list $internal) -}} +{{- range $k, $l := $l.external -}} +{{- if (not (get (fromJson (include "redpanda.KafkaExternal.IsEnabled" (dict "a" (list $l) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $listener := (dict "name" $k "port" ($l.port | int) "address" "0.0.0.0" ) -}} +{{- if (get (fromJson (include "redpanda.Auth.IsSASLEnabled" (dict "a" (list $auth) ))) "r") -}} +{{- $_ := (set $listener "authentication_method" "sasl") -}} +{{- end -}} +{{- $am_10 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_10 "") -}} +{{- $_ := (set $listener "authentication_method" $am_10) -}} +{{- end -}} +{{- $kafka = (concat (default (list ) $kafka) (list $listener)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $kafka) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.KafkaListeners.ListenersTLS" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $kafka := (list ) -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerTLSCfg" (dict "a" (list $tls $l.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $internal) ))) "r") | int) (0 | int)) -}} +{{- $kafka = (concat (default (list ) $kafka) (list $internal)) -}} +{{- end -}} +{{- range $k, $lis := $l.external -}} +{{- if (or (not (get (fromJson (include "redpanda.KafkaExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- $certName := (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $lis.tls $l.tls) ))) "r") -}} +{{- $kafka = (concat (default (list ) $kafka) (list (dict "name" $k "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $certName) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $certName) "require_client_auth" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $lis.tls.requireClientAuth false) ))) "r") "truststore_file" (get (fromJson (include "redpanda.ExternalTLS.TrustStoreFilePath" (dict "a" (list $lis.tls $l.tls $tls) ))) "r") ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $kafka) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.KafkaListeners.TrustStores" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tss := (coalesce nil) -}} +{{- if (and (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $l.tls $tls) ))) "r") (ne (toJson $l.tls.trustStore) "null")) -}} +{{- $tss = (concat (default (list ) $tss) (list $l.tls.trustStore)) -}} +{{- end -}} +{{- range $_, $key := (sortAlpha (keys $l.external)) -}} +{{- $lis := (index $l.external $key) -}} +{{- if (or (or (not (get (fromJson (include "redpanda.KafkaExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) (eq (toJson $lis.tls.trustStore) "null")) -}} +{{- continue -}} +{{- end -}} +{{- $tss = (concat (default (list ) $tss) (list $lis.tls.trustStore)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.KafkaListeners.ConsoleTLS" -}} +{{- $k := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $t := (mustMergeOverwrite (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false ) (dict "enabled" (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $k.tls $tls) ))) "r") )) -}} +{{- if (not $t.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $kafkaPathPrefix := (printf "%s/%s" "/etc/tls/certs" $k.tls.cert) -}} +{{- if (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) $k.tls.cert) ))) "r").caEnabled -}} +{{- $_ := (set $t "caFilepath" (printf "%s/ca.crt" $kafkaPathPrefix)) -}} +{{- else -}} +{{- $_ := (set $t "caFilepath" (printf "%s/tls.crt" $kafkaPathPrefix)) -}} +{{- end -}} +{{- if (not $k.tls.requireClientAuth) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $t "certFilepath" (printf "%s/tls.crt" $kafkaPathPrefix)) -}} +{{- $_ := (set $t "keyFilepath" (printf "%s/tls.key" $kafkaPathPrefix)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.KafkaExternal.IsEnabled" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.enabled true) ))) "r") (gt ($l.port | int) (0 | int)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SchemaRegistryListeners.Listeners" -}} +{{- $sr := (index .a 0) -}} +{{- $saslEnabled := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerCfg" (dict "a" (list ($sr.port | int)) ))) "r") -}} +{{- if $saslEnabled -}} +{{- $_ := (set $internal "authentication_method" "http_basic") -}} +{{- end -}} +{{- $am_11 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $sr.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_11 "") -}} +{{- $_ := (set $internal "authentication_method" $am_11) -}} +{{- end -}} +{{- $result := (list $internal) -}} +{{- range $k, $l := $sr.external -}} +{{- if (not (get (fromJson (include "redpanda.SchemaRegistryExternal.IsEnabled" (dict "a" (list $l) ))) "r")) -}} +{{- continue -}} +{{- end -}} +{{- $listener := (dict "name" $k "port" ($l.port | int) "address" "0.0.0.0" ) -}} +{{- if $saslEnabled -}} +{{- $_ := (set $listener "authentication_method" "http_basic") -}} +{{- end -}} +{{- $am_12 := (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.authenticationMethod "") ))) "r") -}} +{{- if (ne $am_12 "") -}} +{{- $_ := (set $listener "authentication_method" $am_12) -}} +{{- end -}} +{{- $result = (concat (default (list ) $result) (list $listener)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SchemaRegistryListeners.ListenersTLS" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $listeners := (list ) -}} +{{- $internal := (get (fromJson (include "redpanda.createInternalListenerTLSCfg" (dict "a" (list $tls $l.tls) ))) "r") -}} +{{- if (gt ((get (fromJson (include "_shims.len" (dict "a" (list $internal) ))) "r") | int) (0 | int)) -}} +{{- $listeners = (concat (default (list ) $listeners) (list $internal)) -}} +{{- end -}} +{{- range $k, $lis := $l.external -}} +{{- if (or (not (get (fromJson (include "redpanda.SchemaRegistryExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) -}} +{{- continue -}} +{{- end -}} +{{- $certName := (get (fromJson (include "redpanda.ExternalTLS.GetCertName" (dict "a" (list $lis.tls $l.tls) ))) "r") -}} +{{- $listeners = (concat (default (list ) $listeners) (list (dict "name" $k "enabled" true "cert_file" (printf "%s/%s/tls.crt" "/etc/tls/certs" $certName) "key_file" (printf "%s/%s/tls.key" "/etc/tls/certs" $certName) "require_client_auth" (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $lis.tls.requireClientAuth false) ))) "r") "truststore_file" (get (fromJson (include "redpanda.ExternalTLS.TrustStoreFilePath" (dict "a" (list $lis.tls $l.tls $tls) ))) "r") ))) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $listeners) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SchemaRegistryListeners.TrustStores" -}} +{{- $l := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tss := (coalesce nil) -}} +{{- if (and (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $l.tls $tls) ))) "r") (ne (toJson $l.tls.trustStore) "null")) -}} +{{- $tss = (concat (default (list ) $tss) (list $l.tls.trustStore)) -}} +{{- end -}} +{{- range $_, $key := (sortAlpha (keys $l.external)) -}} +{{- $lis := (index $l.external $key) -}} +{{- if (or (or (not (get (fromJson (include "redpanda.SchemaRegistryExternal.IsEnabled" (dict "a" (list $lis) ))) "r")) (not (get (fromJson (include "redpanda.ExternalTLS.IsEnabled" (dict "a" (list $lis.tls $l.tls $tls) ))) "r"))) (eq (toJson $lis.tls.trustStore) "null")) -}} +{{- continue -}} +{{- end -}} +{{- $tss = (concat (default (list ) $tss) (list $lis.tls.trustStore)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $tss) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SchemaRegistryListeners.ConsoleTLS" -}} +{{- $sr := (index .a 0) -}} +{{- $tls := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $t := (mustMergeOverwrite (dict "enabled" false "caFilepath" "" "certFilepath" "" "keyFilepath" "" "insecureSkipTlsVerify" false ) (dict "enabled" (get (fromJson (include "redpanda.InternalTLS.IsEnabled" (dict "a" (list $sr.tls $tls) ))) "r") )) -}} +{{- if (not $t.enabled) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $schemaRegistryPrefix := (printf "%s/%s" "/etc/tls/certs" $sr.tls.cert) -}} +{{- if (get (fromJson (include "redpanda.TLSCertMap.MustGet" (dict "a" (list (deepCopy $tls.certs) $sr.tls.cert) ))) "r").caEnabled -}} +{{- $_ := (set $t "caFilepath" (printf "%s/ca.crt" $schemaRegistryPrefix)) -}} +{{- else -}} +{{- $_ := (set $t "caFilepath" (printf "%s/tls.crt" $schemaRegistryPrefix)) -}} +{{- end -}} +{{- if (not $sr.tls.requireClientAuth) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_ := (set $t "certFilepath" (printf "%s/tls.crt" $schemaRegistryPrefix)) -}} +{{- $_ := (set $t "keyFilepath" (printf "%s/tls.key" $schemaRegistryPrefix)) -}} +{{- $_is_returning = true -}} +{{- (dict "r" $t) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SchemaRegistryExternal.IsEnabled" -}} +{{- $l := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (get (fromJson (include "_shims.ptr_Deref" (dict "a" (list $l.enabled true) ))) "r") (gt ($l.port | int) (0 | int)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TunableConfig.Translate" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- if (eq (toJson $c) "null") -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $result := (dict ) -}} +{{- range $k, $v := $c -}} +{{- if (not (empty $v)) -}} +{{- $_ := (set $result $k $v) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.NodeConfig.Translate" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- range $k, $v := $c -}} +{{- if (not (empty $v)) -}} +{{- $tmp_tuple_12 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.asnumeric" (dict "a" (list $v) ))) "r")) ))) "r") -}} +{{- $ok_13 := $tmp_tuple_12.T2 -}} +{{- if $ok_13 -}} +{{- $_ := (set $result $k $v) -}} +{{- else -}}{{- if (kindIs "bool" $v) -}} +{{- $_ := (set $result $k $v) -}} +{{- else -}} +{{- $_ := (set $result $k (toYaml $v)) -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.ClusterConfig.Translate" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $result := (dict ) -}} +{{- range $k, $v := $c -}} +{{- $tmp_tuple_13 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.typetest" (dict "a" (list "bool" $v false) ))) "r")) ))) "r") -}} +{{- $ok_15 := $tmp_tuple_13.T2 -}} +{{- $b_14 := $tmp_tuple_13.T1 -}} +{{- if $ok_15 -}} +{{- $_ := (set $result $k $b_14) -}} +{{- continue -}} +{{- end -}} +{{- if (not (empty $v)) -}} +{{- $_ := (set $result $k $v) -}} +{{- end -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $result) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretRef.AsSource" -}} +{{- $sr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (mustMergeOverwrite (dict ) (dict "secretKeyRef" (mustMergeOverwrite (dict "key" "" ) (mustMergeOverwrite (dict ) (dict "name" $sr.name )) (dict "key" $sr.key )) ))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.SecretRef.IsValid" -}} +{{- $sr := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and (and (ne (toJson $sr) "null") (not (empty $sr.key))) (not (empty $sr.name)))) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TieredStorageCredentials.AsEnvVars" -}} +{{- $tsc := (index .a 0) -}} +{{- $config := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_14 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $config "cloud_storage_access_key" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $hasAccessKey := $tmp_tuple_14.T2 -}} +{{- $tmp_tuple_15 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $config "cloud_storage_secret_key" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $hasSecretKey := $tmp_tuple_15.T2 -}} +{{- $tmp_tuple_16 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $config "cloud_storage_azure_shared_key" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $hasSharedKey := $tmp_tuple_16.T2 -}} +{{- $envvars := (coalesce nil) -}} +{{- if (and (not $hasAccessKey) (get (fromJson (include "redpanda.SecretRef.IsValid" (dict "a" (list $tsc.accessKey) ))) "r")) -}} +{{- $envvars = (concat (default (list ) $envvars) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_CLOUD_STORAGE_ACCESS_KEY" "valueFrom" (get (fromJson (include "redpanda.SecretRef.AsSource" (dict "a" (list $tsc.accessKey) ))) "r") )))) -}} +{{- end -}} +{{- if (get (fromJson (include "redpanda.SecretRef.IsValid" (dict "a" (list $tsc.secretKey) ))) "r") -}} +{{- if (and (not $hasSecretKey) (not (get (fromJson (include "redpanda.TieredStorageConfig.HasAzureCanaries" (dict "a" (list (deepCopy $config)) ))) "r"))) -}} +{{- $envvars = (concat (default (list ) $envvars) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_CLOUD_STORAGE_SECRET_KEY" "valueFrom" (get (fromJson (include "redpanda.SecretRef.AsSource" (dict "a" (list $tsc.secretKey) ))) "r") )))) -}} +{{- else -}}{{- if (and (not $hasSharedKey) (get (fromJson (include "redpanda.TieredStorageConfig.HasAzureCanaries" (dict "a" (list (deepCopy $config)) ))) "r")) -}} +{{- $envvars = (concat (default (list ) $envvars) (list (mustMergeOverwrite (dict "name" "" ) (dict "name" "REDPANDA_CLOUD_STORAGE_AZURE_SHARED_KEY" "valueFrom" (get (fromJson (include "redpanda.SecretRef.AsSource" (dict "a" (list $tsc.secretKey) ))) "r") )))) -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $envvars) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TieredStorageConfig.HasAzureCanaries" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_17 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $c "cloud_storage_azure_container" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $containerExists := $tmp_tuple_17.T2 -}} +{{- $tmp_tuple_18 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $c "cloud_storage_azure_storage_account" (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $accountExists := $tmp_tuple_18.T2 -}} +{{- $_is_returning = true -}} +{{- (dict "r" (and $containerExists $accountExists)) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TieredStorageConfig.CloudStorageCacheSize" -}} +{{- $c := (index .a 0) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $tmp_tuple_19 := (get (fromJson (include "_shims.compact" (dict "a" (list (get (fromJson (include "_shims.dicttest" (dict "a" (list $c `cloud_storage_cache_size` (coalesce nil)) ))) "r")) ))) "r") -}} +{{- $ok := $tmp_tuple_19.T2 -}} +{{- $value := $tmp_tuple_19.T1 -}} +{{- if (not $ok) -}} +{{- $_is_returning = true -}} +{{- (dict "r" (coalesce nil)) | toJson -}} +{{- break -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $value) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + +{{- define "redpanda.TieredStorageConfig.Translate" -}} +{{- $c := (index .a 0) -}} +{{- $creds := (index .a 1) -}} +{{- range $_ := (list 1) -}} +{{- $_is_returning := false -}} +{{- $config := (merge (dict ) (dict ) $c) -}} +{{- range $_, $envvar := (get (fromJson (include "redpanda.TieredStorageCredentials.AsEnvVars" (dict "a" (list $creds $c) ))) "r") -}} +{{- $key := (lower (substr ((get (fromJson (include "_shims.len" (dict "a" (list "REDPANDA_") ))) "r") | int) -1 $envvar.name)) -}} +{{- $_ := (set $config $key (printf "$%s" $envvar.name)) -}} +{{- end -}} +{{- if $_is_returning -}} +{{- break -}} +{{- end -}} +{{- $size_16 := (get (fromJson (include "redpanda.TieredStorageConfig.CloudStorageCacheSize" (dict "a" (list (deepCopy $c)) ))) "r") -}} +{{- if (ne (toJson $size_16) "null") -}} +{{- $_ := (set $config "cloud_storage_cache_size" ((get (fromJson (include "_shims.resource_Value" (dict "a" (list $size_16) ))) "r") | int64)) -}} +{{- end -}} +{{- $_is_returning = true -}} +{{- (dict "r" $config) | toJson -}} +{{- break -}} +{{- end -}} +{{- end -}} + diff --git a/charts/redpanda/redpanda/5.9.10/templates/connectors/connectors.yaml b/charts/redpanda/redpanda/5.9.10/templates/connectors/connectors.yaml new file mode 100644 index 0000000000..25343f584a --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/connectors/connectors.yaml @@ -0,0 +1,109 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{ if and .Values.connectors.enabled (not .Values.connectors.deployment.create) }} + +{{ $values := .Values }} + +{{/* brokers */}} +{{ $kafkaBrokers := list }} +{{ range (include "seed-server-list" . | mustFromJson) }} + {{ $kafkaBrokers = append $kafkaBrokers (printf "%s:%d" . (int $values.listeners.kafka.port)) }} +{{ end }} + +{{ $connectorsValues := dict + "Values" (dict + "connectors" (dict + "bootstrapServers" (join "," $kafkaBrokers) + "brokerTLS" (dict + "enabled" (include "kafka-internal-tls-enabled" . | fromJson).bool + "ca" (dict + "secretRef" (ternary (printf "%s-default-cert" (include "redpanda.fullname" .)) "" (include "kafka-internal-tls-enabled" . | fromJson).bool) + ) + ) + ) + ) +}} + +{{ $extraVolumes := list }} +{{ $extraVolumeMounts := list }} +{{ $extraEnv := .Values.connectors.deployment.extraEnv }} +{{ $command := list }} +{{ if (include "sasl-enabled" . | fromJson).bool }} + {{ $command = concat $command (list "bash" "-c") }} + {{ $consoleSASLConfig := (printf "set -e; IFS=':' read -r CONNECT_SASL_USERNAME CONNECT_SASL_PASSWORD CONNECT_SASL_MECHANISM < <(grep \"\" $(find /mnt/users/* -print)); CONNECT_SASL_MECHANISM=${CONNECT_SASL_MECHANISM:-%s}; export CONNECT_SASL_USERNAME CONNECT_SASL_PASSWORD CONNECT_SASL_MECHANISM;" ( include "sasl-mechanism" . | lower )) }} + {{ $consoleSASLConfig = cat $consoleSASLConfig " [[ $CONNECT_SASL_MECHANISM == \"SCRAM-SHA-256\" ]] && CONNECT_SASL_MECHANISM=scram-sha-256;" }} + {{ $consoleSASLConfig = cat $consoleSASLConfig " [[ $CONNECT_SASL_MECHANISM == \"SCRAM-SHA-512\" ]] && CONNECT_SASL_MECHANISM=scram-sha-512;" }} + {{ $consoleSASLConfig = cat $consoleSASLConfig " export CONNECT_SASL_MECHANISM;" }} + {{ $consoleSASLConfig = cat $consoleSASLConfig " echo $CONNECT_SASL_PASSWORD > /opt/kafka/connect-password/rc-credentials/password;" }} + {{ $consoleSASLConfig = cat $consoleSASLConfig " exec /opt/kafka/bin/kafka_connect_run.sh" }} + {{ $command = append $command $consoleSASLConfig }} + + {{ $extraVolumes = concat $extraVolumes .Values.connectors.storage.volume }} + + {{ $extraVolumes = append $extraVolumes (dict + "name" (printf "%s-users" (include "redpanda.fullname" .)) + "secret" (dict + "secretName" .Values.auth.sasl.secretRef + ) + )}} + + {{ $extraVolumeMounts = concat $extraVolumeMounts .Values.connectors.storage.volumeMounts }} + + {{ $extraVolumeMounts = append $extraVolumeMounts (dict + "name" (printf "%s-users" (include "redpanda.fullname" .)) + "mountPath" "/mnt/users" + "readOnly" true + )}} + {{ $extraVolumes = append $extraVolumes (dict + "name" (printf "%s-user-password" ((include "redpanda.fullname" .)) | trunc 49) + "emptyDir" (dict) + )}} + {{ $extraVolumeMounts = append $extraVolumeMounts (dict + "name" (printf "%s-user-password" ((include "redpanda.fullname" .)) | trunc 49) + "mountPath" "/opt/kafka/connect-password/rc-credentials" + )}} + {{ $extraEnv = append $extraEnv (dict + "name" "CONNECT_SASL_PASSWORD_FILE" + "value" "rc-credentials/password" + )}} + {{ $connectorsValues := merge $connectorsValues (dict + "Values" (dict + "storage" (dict + "volumeMounts" $extraVolumeMounts + "volume" $extraVolumes + ) + "auth" (dict + "sasl" (dict + "enabled" .Values.auth.sasl.enabled + ) + ) + "deployment" (dict + "command" $command + "extraEnv" $extraEnv + ) + ) + )}} +{{ end }} + +{{ $connectorsValues := merge $connectorsValues (dict "Values" (dict "deployment" (dict "create" (not .Values.connectors.deployment.create)))) }} +{{ $connectorsValues := merge $connectorsValues (dict "Values" (dict "test" (dict "create" (not .Values.connectors.test.create)))) }} +{{ $helmVars := merge $connectorsValues .Subcharts.connectors }} +{{ $helmVars = (dict "Chart" .Subcharts.connectors.Chart "Release" .Release "Values" (merge (dict "AsMap" $helmVars.Values) $helmVars.Values)) }} +{{ include (print .Subcharts.connectors.Template.BasePath "/deployment.yaml") $helmVars }} +--- +{{ include (print .Subcharts.connectors.Template.BasePath "/tests/01-mm2-values.yaml") $helmVars }} +{{ end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/entry-point.yaml b/charts/redpanda/redpanda/5.9.10/templates/entry-point.yaml new file mode 100644 index 0000000000..6cdf646ad6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/entry-point.yaml @@ -0,0 +1,17 @@ +{{- /* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- include "_shims.render-manifest" (list "redpanda.render" .) -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-api-status.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-api-status.yaml new file mode 100644 index 0000000000..330a2c4a4d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-api-status.yaml @@ -0,0 +1,52 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (not (or (include "tls-enabled" . | fromJson).bool (include "sasl-enabled" . | fromJson).bool)) -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-api-status" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + until rpk cluster info \ + --brokers {{ include "redpanda.fullname" . }}-0.{{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.kafka.port }} + do sleep 2 + done + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-auditLogging.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-auditLogging.yaml new file mode 100644 index 0000000000..fea34776fc --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-auditLogging.yaml @@ -0,0 +1,86 @@ +{{/* + Licensed to the Apache Software Foundation (ASF) under one or more + contributor license agreements. See the NOTICE file distributed with + this work for additional information regarding copyright ownership. + The ASF licenses this file to You under the Apache License, Version 2.0 + (the "License"); you may not use this file except in compliance with + the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/}} +{{/* + This feature is gated by having a license, and it must have sasl enabled, we assume these conditions are met + as part of setting auditLogging being enabled. +*/}} +{{- if and .Values.tests.enabled .Values.auditLogging.enabled (include "redpanda-atleast-23-3-0" . | fromJson).bool }} +{{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-audit-logging" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: { { - toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + set -xe + old_setting=${-//[^x]/} + audit_topic_name="_redpanda.audit_log" + expected_partitions={{ .Values.auditLogging.partitions }} + + # sasl configurations + set +x + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + if [[ -n "$old_setting" ]]; then set -x; fi + + # now run the to determine if we have the right results + # should describe topic without error + rpk topic describe ${audit_topic_name} + # should get the expected values + result=$(rpk topic list | grep ${audit_topic_name}) + name=$(echo $result | awk '{print $1}') + partitions=$(echo $result | awk '{print $2}') + if [ "${name}" != "${audit_topic_name}" ]; then + echo "expected topic name does not match" + exit 1 + fi + if [ ${partitions} != ${expected_partitions} ]; then + echo "expected partition size did not match" + exit 1 + fi + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: +{{- toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-connector-via-console.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-connector-via-console.yaml new file mode 100644 index 0000000000..67619a829b --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-connector-via-console.yaml @@ -0,0 +1,166 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled .Values.connectors.enabled .Values.console.enabled }} +{{- $sasl := .Values.auth.sasl }} +{{- $values := .Values }} +{{- $consoleValues := (merge (dict) .Values.console .Subcharts.console.Values) -}} +{{- $consoleDot := dict "Values" (dict "AsMap" $consoleValues) "Release" .Release "Chart" .Subcharts.console.Chart -}} +{{- $connectorsDot := dict "Values" (merge (dict) .Values.connectors .Subcharts.connectors.Values) "Release" .Release "Chart" .Subcharts.connectors.Chart -}} +{{/* brokers */}} +{{- $kafkaBrokers := list }} +{{- range (include "seed-server-list" . | mustFromJson) }} + {{- $kafkaBrokers = append $kafkaBrokers (printf "%s:%s" . ($values.listeners.kafka.port | toString)) }} +{{- end }} +{{- $brokersString := join "," $kafkaBrokers}} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . | trunc 54 }}-test-connectors-via-console + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + test-name: test-connectors-via-console + annotations: + test-name: test-connectors-via-console + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + env: + - name: TLS_ENABLED + value: {{ (include "kafka-internal-tls-enabled" . | fromJson).bool | quote }} + command: + - /bin/bash + - -c + - | + set -xe + + trap connectorsState ERR + + connectorsState () { + echo check connectors expand status + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" $connectorsDot }}:{{ .Values.connectors.connectors.restPort }}/connectors?expand=status + echo check connectors expand info + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" $connectorsDot }}:{{ .Values.connectors.connectors.restPort }}/connectors?expand=info + echo check connector configuration + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" $connectorsDot }}:{{ .Values.connectors.connectors.restPort }}/connectors/$CONNECTOR_NAME + echo check connector topics + curl {{ template "curl-options" . }} http://{{ include "connectors.serviceName" $connectorsDot }}:{{ .Values.connectors.connectors.restPort }}/connectors/$CONNECTOR_NAME/topics + } + + {{- if .Values.auth.sasl.enabled }} + set -e + set +x + + echo "SASL enabled: reading credentials from $(find /etc/secrets/users/* -print)" + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + RPK_USER="${REDPANDA_SASL_USERNAME}" + RPK_PASS="${REDPANDA_SASL_PASSWORD}" + RPK_SASL_MECHANISM="${REDPANDA_SASL_MECHANISM}" + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + + JAAS_CONFIG_SOURCE="\"source.cluster.sasl.jaas.config\": \"org.apache.kafka.common.security.scram.ScramLoginModule required username=\\\\"\"${RPK_USER}\\\\"\" password=\\\\"\"${RPK_PASS}\\\\"\";\"," + JAAS_CONFIG_TARGET="\"target.cluster.sasl.jaas.config\": \"org.apache.kafka.common.security.scram.ScramLoginModule required username=\\\\"\"${RPK_USER}\\\\"\" password=\\\\"\"${RPK_PASS}\\\\"\";\"," + set -x + set +e + {{- end }} + + {{- $testTopic := printf "test-topic-%s" (randNumeric 3) }} + rpk topic create {{ $testTopic }} + rpk topic list + echo "Test message!" | rpk topic produce {{ $testTopic }} + + SECURITY_PROTOCOL=PLAINTEXT + if [[ -n "$RPK_SASL_MECHANISM" && $TLS_ENABLED == "true" ]]; then + SECURITY_PROTOCOL="SASL_SSL" + elif [[ -n "$RPK_SASL_MECHANISM" ]]; then + SECURITY_PROTOCOL="SASL_PLAINTEXT" + elif [[ $TLS_ENABLED == "true" ]]; then + SECURITY_PROTOCOL="SSL" + fi + + CONNECTOR_NAME=mm2-$RANDOM + cat << 'EOF' > /tmp/mm2-conf.json + { + "connectorName": "CONNECTOR_NAME", + "config": { + "connector.class": "org.apache.kafka.connect.mirror.MirrorSourceConnector", + "topics": "{{ $testTopic }}", + "replication.factor": "1", + "tasks.max": "1", + "source.cluster.bootstrap.servers": {{ $brokersString | quote }}, + "target.cluster.bootstrap.servers": {{ $brokersString | quote }}, + "target.cluster.alias": "test-only-redpanda", + "source.cluster.alias": "source", + "key.converter": "org.apache.kafka.connect.converters.ByteArrayConverter", + "value.converter": "org.apache.kafka.connect.converters.ByteArrayConverter", + "source->target.enabled": "true", + "target->source.enabled": "false", + "sync.topic.configs.interval.seconds": "5", + "sync.topics.configs.enabled": "true", + "source.cluster.ssl.truststore.type": "PEM", + "target.cluster.ssl.truststore.type": "PEM", + "source.cluster.ssl.truststore.location": "/opt/kafka/connect-certs/ca/ca.crt", + "target.cluster.ssl.truststore.location": "/opt/kafka/connect-certs/ca/ca.crt", + JAAS_CONFIG_SOURCE + JAAS_CONFIG_TARGET + "source.cluster.security.protocol": "SECURITY_PROTOCOL", + "target.cluster.security.protocol": "SECURITY_PROTOCOL", + "source.cluster.sasl.mechanism": "SASL_MECHANISM", + "target.cluster.sasl.mechanism": "SASL_MECHANISM" + } + } + EOF + + sed -i "s/CONNECTOR_NAME/$CONNECTOR_NAME/g" /tmp/mm2-conf.json + sed -i "s/SASL_MECHANISM/$RPK_SASL_MECHANISM/g" /tmp/mm2-conf.json + sed -i "s/SECURITY_PROTOCOL/$SECURITY_PROTOCOL/g" /tmp/mm2-conf.json + set +x + sed -i "s/JAAS_CONFIG_SOURCE/$JAAS_CONFIG_SOURCE/g" /tmp/mm2-conf.json + sed -i "s/JAAS_CONFIG_TARGET/$JAAS_CONFIG_TARGET/g" /tmp/mm2-conf.json + set -x + + URL=http://{{ get ((include "console.Fullname" (dict "a" (list $consoleDot))) | fromJson) "r" }}:{{ get (fromJson (include "console.ContainerPort" (dict "a" (list $consoleDot) ))) "r" }}/api/kafka-connect/clusters/connectors/connectors + {{/* outputting to /dev/null because the output contains the user password */}} + echo "Creating mm2 connector" + curl {{ template "curl-options" . }} -H 'Content-Type: application/json' "${URL}" -d @/tmp/mm2-conf.json + + rpk topic consume source.{{ $testTopic }} -n 1 + + echo "Destroying mm2 connector" + curl {{ template "curl-options" . }} -X DELETE "${URL}/${CONNECTOR_NAME}" + + rpk topic list + rpk topic delete {{ $testTopic }} source.{{ $testTopic }} mm2-offset-syncs.test-only-redpanda.internal + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-console.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-console.yaml new file mode 100644 index 0000000000..aeef1117ac --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-console.yaml @@ -0,0 +1,49 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled .Values.console.enabled -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-console" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + curl {{ template "curl-options" . }} http://{{ include "redpanda.fullname" . }}-console.{{ .Release.Namespace }}.svc:{{ (get (fromJson (include "console.ContainerPort" (dict "a" (list (dict "Values" (dict "AsMap" .Values.console)) )))) "r" ) }}/api/cluster + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-internal-external-tls-secrets.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-internal-external-tls-secrets.yaml new file mode 100644 index 0000000000..53d75bb1ba --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-internal-external-tls-secrets.yaml @@ -0,0 +1,122 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (include "tls-enabled" . | fromJson).bool ( eq .Values.external.type "NodePort" ) }} + {{- $values := .Values }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-internal-externals-cert-secrets + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - bash + - -c + - | + set -x + + retry() { + local retries="$1" + local command="$2" + + # Run the command, and save the exit code + bash -c $command + local exit_code=$? + + # If the exit code is non-zero (i.e. command failed), and we have not + # reached the maximum number of retries, run the command again + if [[ $exit_code -ne 0 && $retries -gt 0 ]]; then + retry $(($retries - 1)) "$command" + else + # Return the exit code from the command + return $exit_code + fi + } + + {{- range $name, $cert := $values.tls.certs }} + {{- if $cert.secretRef }} + echo testing cert: {{ $name | quote }} + + {{- if eq $cert.secretRef.name "internal-tls-secret" }} + echo "---> testing internal tls" + retry 5 'openssl s_client -verify_return_error -prexit + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key + -connect {{ include "admin-api-urls" $ }}' + {{- end }} + + {{- if eq $cert.secretRef.name "external-tls-secret" }} + echo "---> testing external tls" + + {{- if eq $values.listeners.kafka.external.default.tls.cert $name }} + echo "-----> testing external tls: kafka api" + {{- $port := ( first $values.listeners.kafka.external.default.advertisedPorts ) }} + retry 5 'openssl s_client -verify_return_error -prexit + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key + -connect {{ $values.external.domain }}:{{ $port }}' + {{- end }} + + {{- if and (eq $values.listeners.schemaRegistry.external.default.tls.cert $name) (include "redpanda-22-2-x-without-sasl" $ | fromJson).bool }} + echo "-----> testing external tls: schema registry" + {{- $port := ( first $values.listeners.schemaRegistry.external.default.advertisedPorts ) }} + retry 5 'openssl s_client -verify_return_error -prexit + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key + -connect {{ $values.external.domain }}:{{ $port }}' + {{- end }} + + {{- if and (eq $values.listeners.http.external.default.tls.cert $name) (include "redpanda-22-2-x-without-sasl" $ | fromJson).bool }} + echo "-----> testing external tls: http api" + {{- $port := ( first $values.listeners.http.external.default.advertisedPorts ) }} + retry 5 'openssl s_client -verify_return_error -prexit + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key + -connect {{ $values.external.domain }}:{{ $port }}' + {{- end }} + + {{- end }} + echo "----" + + {{- end }} + {{- end }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-internal-tls-status.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-internal-tls-status.yaml new file mode 100644 index 0000000000..dcfc02cbdc --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-internal-tls-status.yaml @@ -0,0 +1,62 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (include "kafka-internal-tls-enabled" . | fromJson).bool (not (include "sasl-enabled" . | fromJson).bool) -}} + {{- $service := .Values.listeners.kafka -}} + {{- $cert := get .Values.tls.certs $service.tls.cert -}} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-kafka-internal-tls-status + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + until rpk cluster info \ + --brokers {{ include "redpanda.fullname" .}}-0.{{ include "redpanda.internal.domain" . }}:{{ $service.port }} \ + --tls-enabled \ + {{- if $cert.caEnabled }} + --tls-truststore /etc/tls/certs/{{ $service.tls.cert }}/ca.crt + {{- else }} + {{- /* This is a required field so we use the default in the redpanda debian container */}} + --tls-truststore /etc/ssl/certs/ca-certificates.crt + {{- end }} + do sleep 2 + done + resources: {{ toYaml .Values.statefulset.resources | nindent 12 }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-nodelete.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-nodelete.yaml new file mode 100644 index 0000000000..9b5fe4237e --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-nodelete.yaml @@ -0,0 +1,100 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (dig "kafka_nodelete_topics" "[]" $.Values.config.cluster) }} +{{- $noDeleteTopics := .Values.config.cluster.kafka_nodelete_topics }} +{{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-kafka-nodelete + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} +{{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + env: + - name: REDPANDA_BROKERS + value: "{{ include "redpanda.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain | trimSuffix "." }}:{{ .Values.listeners.kafka.port }}" + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + set -e +{{- $cloudStorageFlags := "" }} +{{- if (include "storage-tiered-config" .|fromJson).cloud_storage_enabled }} + {{- $cloudStorageFlags = "-c retention.bytes=80 -c segment.bytes=40 -c redpanda.remote.read=true -c redpanda.remote.write=true"}} +{{- end }} +{{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + if [[ -n "$old_setting" ]]; then set -x; fi +{{- end }} + + exists=$(rpk topic list | grep my_sample_topic | awk '{print $1}') + if [[ "$exists" != "my_sample_topic" ]]; then + until rpk topic create my_sample_topic {{ $cloudStorageFlags }} + do sleep 2 + done + fi + + {{- range $i := until 100 }} + echo "Pandas are awesome!" | rpk topic produce my_sample_topic + {{- end }} + sleep 2 + rpk topic consume my_sample_topic -n 1 | grep "Pandas are awesome!" + + # now check if we can delete the topic (we should not) + rpk topic delete my_sample_topic + + {{- if has "my_sample_topic" $noDeleteTopics }} + result=$(rpk topic list | grep my_sample_topic | awk '{print $1}') + if [[ "$result" != "my_sample_topic" ]]; then + echo "topic should not have been deleted" + exit 1 + fi + {{- end }} + + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: {{ toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-produce-consume.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-produce-consume.yaml new file mode 100644 index 0000000000..d8f0ee7518 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-produce-consume.yaml @@ -0,0 +1,83 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if .Values.tests.enabled }} +{{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-kafka-produce-consume + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} +{{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + env: + - name: REDPANDA_BROKERS + value: "{{ include "redpanda.fullname" . }}.{{ .Release.Namespace }}.svc.{{ .Values.clusterDomain | trimSuffix "." }}:{{ .Values.listeners.kafka.port }}" + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + set -e +{{- $cloudStorageFlags := "" }} +{{- if (include "storage-tiered-config" .|fromJson).cloud_storage_enabled }} + {{- $cloudStorageFlags = "-c retention.bytes=80 -c segment.bytes=40 -c redpanda.remote.read=true -c redpanda.remote.write=true"}} +{{- end }} +{{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + if [[ -n "$old_setting" ]]; then set -x; fi +{{- end }} + until rpk topic create produce.consume.test.$POD_NAME {{ $cloudStorageFlags }} + do sleep 2 + done + {{- range $i := until 100 }} + echo "Pandas are awesome!" | rpk topic produce produce.consume.test.$POD_NAME + {{- end }} + sleep 2 + rpk topic consume produce.consume.test.$POD_NAME -n 1 | grep "Pandas are awesome!" + rpk topic delete produce.consume.test.$POD_NAME + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: {{ toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-sasl-status.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-sasl-status.yaml new file mode 100644 index 0000000000..0519c44bba --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-kafka-sasl-status.yaml @@ -0,0 +1,79 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (include "sasl-enabled" . | fromJson).bool }} +{{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-kafka-sasl-status" + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + set -xe + +{{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + if [[ -n "$old_setting" ]]; then set -x; fi +{{- end }} + + until rpk acl user delete myuser + do sleep 2 + done + sleep 3 + + {{ include "rpk-cluster-info" $ }} + {{ include "rpk-acl-user-create" $ }} + {{ include "rpk-acl-create" $ }} + sleep 3 + {{ include "rpk-topic-create" $ }} + {{ include "rpk-topic-describe" $ }} + {{ include "rpk-topic-delete" $ }} + rpk acl user delete myuser + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: +{{- toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-license-with-console.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-license-with-console.yaml new file mode 100644 index 0000000000..1edf7a3507 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-license-with-console.yaml @@ -0,0 +1,61 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + +http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (include "is-licensed" . | fromJson).bool .Values.console.enabled }} +{{- $consolePort := (get (fromJson (include "console.ContainerPort" (dict "a" (list (dict "Values" (dict "AsMap" .Values.console)) )))) "r" ) }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-license-with-console" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: + runAsUser: 65535 + runAsGroup: 65535 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: mintel/docker-alpine-bash-curl-jq:latest + command: [ "/bin/bash", "-c" ] + args: + - | + echo "testing that we do NOT have an open source license" + set -xe + + max_iteration=10 + curl -vm3 --fail --retry "120" --retry-max-time "120" http://{{ include "redpanda.fullname" . }}-console.{{ .Release.Namespace }}.svc:{{$consolePort}}/api/cluster/overview | jq . + type=$(curl -svm3 --fail --retry "120" --retry-max-time "120" http://{{ include "redpanda.fullname" . }}-console.{{ .Release.Namespace }}.svc:{{$consolePort}}/api/cluster/overview | jq -r .console.license.type) + while [[ $max_iteration -gt 0 && ("$type" == "open_source" || "$type" == "") ]]; do + max_iteration=$(( max_iteration - 1 )) + type=$(curl -svm3 --fail --retry "120" --retry-max-time "120" http://{{ include "redpanda.fullname" . }}-console.{{ .Release.Namespace }}.svc:{{$consolePort}}/api/cluster/overview | jq -r .console.license.type) + done + if [[ "$type" == "open_source" || "$type" == "" ]]; then + curl -svm3 --fail --retry "120" --retry-max-time "120" http://{{ include "redpanda.fullname" . }}-console.{{ .Release.Namespace }}.svc:{{$consolePort}}/api/cluster/overview | jq . + exit 1 + fi + set +x + echo "license test passed." +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-lifecycle-scripts.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-lifecycle-scripts.yaml new file mode 100644 index 0000000000..5c72e1d9fb --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-lifecycle-scripts.yaml @@ -0,0 +1,66 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if .Values.tests.enabled }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-lifecycle" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + env: + - name: SERVICE_NAME + value: {{ include "redpanda.fullname" . }}-0 + command: + - /bin/timeout + - "{{ mul .Values.statefulset.terminationGracePeriodSeconds 2 }}" + - bash + - -xec + - | + /bin/timeout -v {{ div .Values.statefulset.terminationGracePeriodSeconds 2 }} bash -x /var/lifecycle/preStop.sh + ls -l /tmp/preStop* + test -f /tmp/preStopHookStarted + test -f /tmp/preStopHookFinished + + /bin/timeout -v {{ div .Values.statefulset.terminationGracePeriodSeconds 2 }} bash -x /var/lifecycle/postStart.sh + ls -l /tmp/postStart* + test -f /tmp/postStartHookStarted + test -f /tmp/postStartHookFinished + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + - name: lifecycle-scripts + mountPath: /var/lifecycle + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} + - name: lifecycle-scripts + secret: + secretName: {{ (include "redpanda.fullname" . | trunc 50 ) }}-sts-lifecycle + defaultMode: 0o775 + {{- end }} \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-loadbalancer-tls.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-loadbalancer-tls.yaml new file mode 100644 index 0000000000..4db3523d2b --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-loadbalancer-tls.yaml @@ -0,0 +1,173 @@ +{{/* + Licensed to the Apache Software Foundation (ASF) under one or more + contributor license agreements. See the NOTICE file distributed with + this work for additional information regarding copyright ownership. + The ASF licenses this file to You under the Apache License, Version 2.0 + (the "License"); you may not use this file except in compliance with + the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */}} +{{- if and .Values.tests.enabled .Values.tls.enabled ( eq .Values.external.type "LoadBalancer" ) -}} + {{- $values := .Values }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-loadbalancer-tls + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + serviceAccountName: test-loadbalancer-tls-redpanda + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: mintel/docker-alpine-bash-curl-jq:latest + command: + - bash + - -c + - | + set -x + export APISERVER=https://kubernetes.default.svc + export SERVICEACCOUNT=/var/run/secrets/kubernetes.io/serviceaccount + export NAMESPACE=$(cat ${SERVICEACCOUNT}/namespace) + export TOKEN=$(cat ${SERVICEACCOUNT}/token) + export CACERT=${SERVICEACCOUNT}/ca.crt + + ip_list="" + + replicas={{ .Values.statefulset.replicas }} + if [ "${replicas}" -lt "1" ]; then + echo "replicas cannot be less than 1" + exit 1 + fi + + range=$(expr $replicas - 1) + ordinal_list=$(seq 0 $range) + + set -e + + for i in $ordinal_list + do + POD_DESC=$(curl --cacert ${CACERT} --header "Authorization: Bearer ${TOKEN}" \ + -X GET ${APISERVER}/api/v1/namespaces/{{ .Release.Namespace }}/services/lb-{{ template "redpanda.fullname" . }}-$i) + ip=$(echo $POD_DESC | jq -r .status.loadBalancer.ingress[0].ip ) + ip_list="$ip $ip_list" + done + + echo test will be run against $ip_list + echo testing LoadBalancer connectivity + + {{- range $name, $cert := $values.tls.certs }} + {{- if $cert.secretRef }} + {{- if eq $cert.secretRef.name "external-tls-secret" }} + echo "---> testing external tls" + + {{- if eq $values.listeners.kafka.external.default.tls.cert $name }} + echo "-----> testing external tls: kafka api" + {{- $port := ( first $values.listeners.kafka.external.default.advertisedPorts ) }} + + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled -}} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end -}} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key -connect $ip:{{ $port }} + done + {{- end }} + + {{- if (include "redpanda-22-2-x-without-sasl" $ | fromJson).bool }} + {{- if eq $values.listeners.schemaRegistry.external.default.tls.cert $name }} + echo "-----> testing external tls: schema registry" + {{- $port := ( first $values.listeners.schemaRegistry.external.default.advertisedPorts ) }} + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled -}} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end -}} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key -connect $ip:{{ $port }} + done + {{- end }} + + {{- if eq $values.listeners.http.external.default.tls.cert $name }} + echo "-----> testing external tls: http api" + {{- $port := ( first $values.listeners.http.external.default.advertisedPorts ) }} + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled -}} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end -}} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key -connect $ip:{{ $port }} + done + {{- end }} + {{- end }} + + {{- end }} + {{- end }} + {{- end }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: test-loadbalancer-tls-redpanda + annotations: + helm.sh/hook-weight: "-100" + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: test-loadbalancer-tls-redpanda + annotations: + helm.sh/hook-weight: "-100" + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: test-loadbalancer-tls-redpanda +subjects: + - kind: ServiceAccount + name: test-loadbalancer-tls-redpanda + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: test-loadbalancer-tls-redpanda + annotations: + helm.sh/hook-weight: "-100" + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +rules: + - apiGroups: + - "" + resources: + - pods + - services + verbs: + - get + +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-nodeport-tls.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-nodeport-tls.yaml new file mode 100644 index 0000000000..4310eaf3a9 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-nodeport-tls.yaml @@ -0,0 +1,173 @@ +{{/* + Licensed to the Apache Software Foundation (ASF) under one or more + contributor license agreements. See the NOTICE file distributed with + this work for additional information regarding copyright ownership. + The ASF licenses this file to You under the Apache License, Version 2.0 + (the "License"); you may not use this file except in compliance with + the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */}} +{{- if and .Values.tests.enabled .Values.tls.enabled ( eq .Values.external.type "NodePort" ) -}} + {{- $values := .Values }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-nodeport-tls + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + serviceAccountName: test-nodeport-tls-redpanda-no-a-test + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: mintel/docker-alpine-bash-curl-jq:latest + command: + - bash + - -c + - | + set -x + export APISERVER=https://kubernetes.default.svc + export SERVICEACCOUNT=/var/run/secrets/kubernetes.io/serviceaccount + export NAMESPACE=$(cat ${SERVICEACCOUNT}/namespace) + export TOKEN=$(cat ${SERVICEACCOUNT}/token) + export CACERT=${SERVICEACCOUNT}/ca.crt + + ip_list="" + + replicas={{ .Values.statefulset.replicas }} + if [ "${replicas}" -lt "1" ]; then + echo "replicas cannot be less than 1" + exit 1 + fi + + range=$(expr $replicas - 1) + ordinal_list=$(seq 0 $range) + + set -e + + for i in $ordinal_list + do + POD_DESC=$(curl --cacert ${CACERT} --header "Authorization: Bearer ${TOKEN}" \ + -X GET ${APISERVER}/api/v1/namespaces/{{ .Release.Namespace }}/pods/{{ template "redpanda.fullname" . }}-$i) + ip=$(echo $POD_DESC | jq -r .status.hostIP ) + ip_list="$ip $ip_list" + done + + echo test will be run against $ip_list + echo testing NodePort connectivity + {{- range $name, $cert := $values.tls.certs }} + {{- if $cert.secretRef }} + {{- if eq $cert.secretRef.name "external-tls-secret" }} + echo "---> testing external tls" + + {{- if eq $values.listeners.kafka.external.default.tls.cert $name }} + echo "-----> testing external tls: kafka api" + {{- $port := ( first $values.listeners.kafka.external.default.advertisedPorts ) }} + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key \ + -connect ${ip}:{{ $port }} + done + {{- end }} + + {{- if (include "redpanda-22-2-x-without-sasl" $ | fromJson).bool }} + {{- if eq $values.listeners.schemaRegistry.external.default.tls.cert $name }} + echo "-----> testing external tls: schema registry" + {{- $port := ( first $values.listeners.schemaRegistry.external.default.advertisedPorts ) }} + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key \ + -connect ${ip}:{{ $port }} + done + {{- end }} + + {{- if eq $values.listeners.http.external.default.tls.cert $name }} + echo "-----> testing external tls: http api" + {{- $port := ( first $values.listeners.http.external.default.advertisedPorts ) }} + for ip in $ip_list + do + openssl s_client -verify_return_error -prexit \ + {{- if $cert.caEnabled }} + -CAfile {{ printf "/etc/tls/certs/%s" $name }}/ca.crt \ + {{- end }} + -key {{ printf "/etc/tls/certs/%s" $name }}/tls.key \ + -connect ${ip}:{{ $port }} + done + {{- end }} + {{- end }} + + {{- end }} + {{- end }} + {{- end }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: test-nodeport-tls-redpanda-no-a-test + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation + helm.sh/hook-weight: "-100" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: test-nodeport-tls-redpanda-no-a-test + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation + helm.sh/hook-weight: "-100" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: test-nodeport-tls-redpanda-no-a-test +subjects: + - kind: ServiceAccount + name: test-nodeport-tls-redpanda-no-a-test + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: test-nodeport-tls-redpanda-no-a-test + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation + helm.sh/hook-weight: "-100" +rules: + - apiGroups: + - "" + resources: + - pods + - services + verbs: + - get +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-internal-tls-status.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-internal-tls-status.yaml new file mode 100644 index 0000000000..4cb6aaa0f6 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-internal-tls-status.yaml @@ -0,0 +1,81 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (include "http-internal-tls-enabled" . | fromJson).bool .Values.listeners.http.enabled (include "redpanda-22-2-x-without-sasl" . | fromJson).bool -}} + {{- $service := .Values.listeners.http -}} + {{- $cert := get .Values.tls.certs $service.tls.cert -}} + {{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-pandaproxy-internal-tls-status + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: [ "/bin/bash", "-c" ] + args: + - | + {{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=":" read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + RPK_USER="${RPK_USER:-${REDPANDA_SASL_USERNAME}}" + RPK_PASS="${RPK_PASS:-${REDPANDA_SASL_PASSWORD}}" + if [[ -n "$old_setting" ]]; then set -x; fi + {{- end }} + + curl -svm3 --fail --retry "120" --retry-max-time "120" --retry-all-errors --ssl-reqd \ + {{- if or (include "sasl-enabled" .|fromJson).bool .Values.listeners.http.authenticationMethod }} + -u ${RPK_USER}:${RPK_PASS} \ + {{- end }} + {{- if $cert.caEnabled }} + --cacert /etc/tls/certs/{{ $service.tls.cert }}/ca.crt \ + {{- end }} + https://{{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.http.port }}/brokers + + curl -svm3 --fail --retry "120" --retry-max-time "120" --retry-all-errors --ssl-reqd \ + {{- if or (include "sasl-enabled" .|fromJson).bool .Values.listeners.http.authenticationMethod }} + -u ${RPK_USER}:${RPK_PASS} \ + {{- end }} + {{- if $cert.caEnabled }} + --cacert /etc/tls/certs/{{ $service.tls.cert }}/ca.crt \ + {{- end }} + https://{{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.http.port }}/topics + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: {{ toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end -}} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-status.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-status.yaml new file mode 100644 index 0000000000..4f5ee6bb71 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-pandaproxy-status.yaml @@ -0,0 +1,72 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if and .Values.tests.enabled (not (include "http-internal-tls-enabled" . | fromJson).bool) .Values.listeners.http.enabled (include "redpanda-22-2-x-without-sasl" . | fromJson).bool -}} + {{- $sasl := .Values.auth.sasl }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-pandaproxy-status" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: [ "/bin/bash", "-c" ] + args: + - | + {{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=: read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + RPK_USER="${RPK_USER:-${REDPANDA_SASL_USERNAME}}" + RPK_PASS="${RPK_PASS:-${REDPANDA_SASL_PASSWORD}}" + if [[ -n "$old_setting" ]]; then set -x; fi + {{- end }} + + curl {{ template "curl-options" . }} \ + {{- if or (include "sasl-enabled" .|fromJson).bool .Values.listeners.http.authenticationMethod }} + -u ${RPK_USER}:${RPK_PASS} \ + {{- end }} + http://{{ include "redpanda.servicename" . }}:{{ .Values.listeners.http.port }}/brokers + + curl {{ template "curl-options" . }} \ + {{- if or (include "sasl-enabled" .|fromJson).bool .Values.listeners.http.authenticationMethod }} + -u ${RPK_USER}:${RPK_PASS} \ + {{- end }} + http://{{ include "redpanda.servicename" . }}:{{ .Values.listeners.http.port }}/topics + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-prometheus-targets.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-prometheus-targets.yaml new file mode 100644 index 0000000000..81f83a34e2 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-prometheus-targets.yaml @@ -0,0 +1,84 @@ +{{/* + Licensed to the Apache Software Foundation (ASF) under one or more + contributor license agreements. See the NOTICE file distributed with + this work for additional information regarding copyright ownership. + The ASF licenses this file to You under the Apache License, Version 2.0 + (the "License"); you may not use this file except in compliance with + the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */}} + +{{- if and .Values.tests.enabled .Values.monitoring.enabled }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-prometheus-targets" + namespace: {{ .Release.Namespace | quote }} + labels: + {{- with include "full.labels" . }} + {{- . | nindent 4 }} + {{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: registry.gitlab.com/gitlab-ci-utils/curl-jq:latest + command: [ "/bin/bash", "-c" ] + args: + - | + set -xe + + HEALTHY=$( curl {{ template "curl-options" . }} http://prometheus-operated.prometheus.svc.cluster.local:9090/-/healthy) + if [ $HEALTHY != 200 ]; then + echo "prometheus is not healthy, exiting" + exit 1 + fi + + echo "prometheus is healthy, checking if ready..." + + READY=$( curl {{ template "curl-options" . }} http://prometheus-operated.prometheus.svc.cluster.local:9090/-/ready) + if [ $READY != 200 ]; then + echo "prometheus is not ready, exiting" + exit 1 + fi + + echo "prometheus is ready, requesting target information..." + + + curl_prometheus() { + + # Run the command, and save the exit code + # from: https://prometheus.io/docs/prometheus/latest/querying/api/ + local RESULT=$( curl {{ template "curl-options" . }} http://prometheus-operated.prometheus.svc.cluster.local:9090/api/v1/targets?scrapePool=serviceMonitor/{{ .Release.Namespace }}/{{ include "redpanda.fullname" . }}/0 | jq '.data.activeTargets[].health | select(. == "up")' | wc -l ) + + echo $RESULT + } + for d in $(seq 1 30); do + RESULT=$(curl_prometheus) + if [ $RESULT == {{ .Values.statefulset.replicas }} ]; then + break + fi + sleep 15 + done + + set +x + if [ $RESULT != {{ .Values.statefulset.replicas }} ]; then + curl --fail http://prometheus-operated.prometheus.svc.cluster.local:9090/api/v1/targets?scrapePool=serviceMonitor/{{ .Release.Namespace }}/{{ include "redpanda.fullname" . }}/0 | jq . + echo "the number of targets unexpected; got ${RESULT} targets 'up', but was expecting {{ .Values.statefulset.replicas }}" + exit 1 + fi +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-rack-awareness.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-rack-awareness.yaml new file mode 100644 index 0000000000..82a31937f5 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-rack-awareness.yaml @@ -0,0 +1,61 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} +{{- if .Values.tests.enabled }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-rack-awareness + namespace: {{ .Release.Namespace | quote }} +{{- with include "full.labels" . }} + labels: {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} +{{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} +{{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /bin/bash + - -c + - | + set -e +{{- if and .Values.rackAwareness.enabled (include "redpanda-atleast-22-3-0" . | fromJson).bool }} + curl {{ template "curl-options" . }} \ + {{- if (include "tls-enabled" . | fromJson).bool }} + {{- if (dig "default" "caEnabled" false .Values.tls.certs) }} + --cacert "/etc/tls/certs/default/ca.crt" \ + {{- end }} + https://{{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.admin.port }}/v1/node_config | grep '"rack":"rack[1-4]"' + {{- else }} + http://{{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.admin.port }}/v1/node_config | grep '"rack":"rack[1-4]"' + {{- end }} +{{- end }} + + rpk redpanda admin config print --host {{ include "redpanda.internal.domain" . }}:{{ .Values.listeners.admin.port }} | grep '"enable_rack_awareness": {{ .Values.rackAwareness.enabled }}' + + rpk cluster config get enable_rack_awareness + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-rpk-debug-bundle.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-rpk-debug-bundle.yaml new file mode 100644 index 0000000000..3230f08817 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-rpk-debug-bundle.yaml @@ -0,0 +1,104 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} + +{{/* + +This test currently fails because of a bug where when multiple containers exist +The api returns an error. We should be requesting logs from each container. + + +{{- if and .Values.tests.enabled .Values.rbac.enabled (include "redpanda-atleast-23-1-1" .|fromJson).bool -}} + {{- $sasl := .Values.auth.sasl }} + {{- $useSaslSecret := and $sasl.enabled (not (empty $sasl.secretRef )) }} + + +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "redpanda.fullname" . }}-test-rpk-debug-bundle + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + affinity: + podAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + statefulset.kubernetes.io/pod-name: {{ include "redpanda.fullname" . }}-0 + topologyKey: kubernetes.io/hostname + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + initContainers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository}}:{{ template "redpanda.tag" . }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + - name: shared-data + mountPath: /usr/share/redpanda/test + - name: datadir + mountPath: /var/lib/redpanda/data + command: + - /bin/bash + - -c + - | + set -e + {{- if .Values.auth.sasl.enabled }} + old_setting=${-//[^x]/} + set +x + IFS=: read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + if [[ -n "$old_setting" ]]; then set -x; fi + {{- end }} + rpk debug bundle -o /usr/share/redpanda/test/debug-test.zip -n {{ .Release.Namespace }} + containers: + - name: {{ template "redpanda.name" . }}-tester + image: busybox:latest + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + - name: shared-data + mountPath: /test + command: + - /bin/ash + - -c + - | + set -e + unzip /test/debug-test.zip -d /tmp/bundle + + test -f /tmp/bundle/logs/{{ .Release.Namespace }}-0.txt + test -f /tmp/bundle/logs/{{ .Release.Namespace }}-1.txt + test -f /tmp/bundle/logs/{{ .Release.Namespace }}-2.txt + + test -d /tmp/bundle/controller + + test -f /tmp/bundle/k8s/pods.json + test -f /tmp/bundle/k8s/configmaps.json + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end -}} +*/}} \ No newline at end of file diff --git a/charts/redpanda/redpanda/5.9.10/templates/tests/test-sasl-updated.yaml b/charts/redpanda/redpanda/5.9.10/templates/tests/test-sasl-updated.yaml new file mode 100644 index 0000000000..5f61be552e --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/templates/tests/test-sasl-updated.yaml @@ -0,0 +1,71 @@ +{{/* +Licensed to the Apache Software Foundation (ASF) under one or more +contributor license agreements. See the NOTICE file distributed with +this work for additional information regarding copyright ownership. +The ASF licenses this file to You under the Apache License, Version 2.0 +(the "License"); you may not use this file except in compliance with +the License. You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/}} + +{{- if and .Values.tests.enabled (include "sasl-enabled" . | fromJson).bool (eq .Values.auth.sasl.secretRef "some-users") -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "redpanda.fullname" . }}-test-update-sasl-users" + namespace: {{ .Release.Namespace | quote }} + labels: +{{- with include "full.labels" . }} + {{- . | nindent 4 }} +{{- end }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": before-hook-creation +spec: + restartPolicy: Never + securityContext: {{ include "pod-security-context" . | nindent 4 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ template "redpanda.name" . }} + image: {{ .Values.image.repository }}:{{ template "redpanda.tag" . }} + command: + - /usr/bin/timeout + - "120" + - bash + - -c + - | + set -e + IFS=: read -r {{ include "rpk-sasl-environment-variables" . }} < <(grep "" $(find /etc/secrets/users/* -print)) + {{- if (include "redpanda-atleast-23-2-1" . | fromJson).bool }} + RPK_SASL_MECHANISM=${RPK_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- else }} + REDPANDA_SASL_MECHANISM=${REDPANDA_SASL_MECHANISM:-{{ .Values.auth.sasl.mechanism | upper }}} + {{- end }} + export {{ include "rpk-sasl-environment-variables" . }} + + set -x + + # check that the users list did update + ready_result_exit_code=1 + while [[ ${ready_result_exit_code} -ne 0 ]]; do + ready_result=$(rpk acl user list | grep anotheranotherme 2>&1) && ready_result_exit_code=$? + sleep 2 + done + + # check that sasl is not broken + {{ include "rpk-cluster-info" $ }} + volumeMounts: {{ include "default-mounts" . | nindent 8 }} + resources: +{{- toYaml .Values.statefulset.resources | nindent 12 }} + securityContext: {{ include "container-security-context" . | nindent 8 }} + volumes: {{ include "default-volumes" . | nindent 4 }} +{{- end }} diff --git a/charts/redpanda/redpanda/5.9.10/values.schema.json b/charts/redpanda/redpanda/5.9.10/values.schema.json new file mode 100644 index 0000000000..99155f8f0d --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/values.schema.json @@ -0,0 +1,12882 @@ +{ + "$id": "https://github.com/redpanda-data/helm-charts/charts/redpanda/values", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "description": "DO NOT EDIT!. This file was generated by ./cmd/genschema/genschema.go", + "properties": { + "affinity": { + "properties": { + "nodeAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "preference": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "properties": { + "nodeSelectorTerms": { + "oneOf": [ + { + "items": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "podAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "podAntiAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "auditLogging": { + "properties": { + "clientMaxBufferSize": { + "type": "integer" + }, + "enabled": { + "type": "boolean" + }, + "enabledEventTypes": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "excludedPrincipals": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "excludedTopics": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "listener": { + "type": "string" + }, + "partitions": { + "type": "integer" + }, + "queueDrainIntervalMs": { + "type": "integer" + }, + "queueMaxBufferSizePerShard": { + "type": "integer" + }, + "replicationFactor": { + "oneOf": [ + { + "type": "integer" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "auth": { + "properties": { + "sasl": { + "properties": { + "bootstrapUser": { + "properties": { + "mechanism": { + "pattern": "^(SCRAM-SHA-512|SCRAM-SHA-256)$", + "type": "string" + }, + "name": { + "type": "string" + }, + "password": { + "type": "string" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "mechanism": { + "type": "string" + }, + "secretRef": { + "type": "string" + }, + "users": { + "oneOf": [ + { + "items": { + "properties": { + "mechanism": { + "pattern": "^(SCRAM-SHA-512|SCRAM-SHA-256)$", + "type": "string" + }, + "name": { + "type": "string" + }, + "password": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "required": [ + "enabled" + ], + "type": "object" + } + }, + "required": [ + "sasl" + ], + "type": "object" + }, + "clusterDomain": { + "type": "string" + }, + "commonLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "config": { + "properties": { + "cluster": { + "type": "object" + }, + "node": { + "type": "object" + }, + "pandaproxy_client": { + "properties": { + "consumer_heartbeat_interval_ms": { + "type": "integer" + }, + "consumer_rebalance_timeout_ms": { + "type": "integer" + }, + "consumer_request_max_bytes": { + "type": "integer" + }, + "consumer_request_timeout_ms": { + "type": "integer" + }, + "consumer_session_timeout_ms": { + "type": "integer" + }, + "produce_batch_delay_ms": { + "type": "integer" + }, + "produce_batch_record_count": { + "type": "integer" + }, + "produce_batch_size_bytes": { + "type": "integer" + }, + "retries": { + "type": "integer" + }, + "retry_base_backoff_ms": { + "type": "integer" + } + }, + "type": "object" + }, + "rpk": { + "type": "object" + }, + "schema_registry_client": { + "properties": { + "consumer_heartbeat_interval_ms": { + "type": "integer" + }, + "consumer_rebalance_timeout_ms": { + "type": "integer" + }, + "consumer_request_max_bytes": { + "type": "integer" + }, + "consumer_request_timeout_ms": { + "type": "integer" + }, + "consumer_session_timeout_ms": { + "type": "integer" + }, + "produce_batch_delay_ms": { + "type": "integer" + }, + "produce_batch_record_count": { + "type": "integer" + }, + "produce_batch_size_bytes": { + "type": "integer" + }, + "retries": { + "type": "integer" + }, + "retry_base_backoff_ms": { + "type": "integer" + } + }, + "type": "object" + }, + "tunable": { + "additionalProperties": true, + "properties": { + "group_initial_rebalance_delay": { + "type": "integer" + }, + "log_retention_ms": { + "type": "integer" + } + }, + "type": "object" + } + }, + "required": [ + "cluster", + "node", + "tunable" + ], + "type": "object" + }, + "connectors": { + "properties": { + "connectors": { + "properties": { + "fullnameOverwrite": { + "type": "string" + }, + "restPort": { + "type": "integer" + } + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + } + }, + "type": "object" + }, + "console": { + "properties": { + "affinity": { + "properties": { + "nodeAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "preference": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "properties": { + "nodeSelectorTerms": { + "oneOf": [ + { + "items": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "podAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "podAntiAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "automountServiceAccountToken": { + "type": "boolean" + }, + "autoscaling": { + "properties": { + "enabled": { + "type": "boolean" + }, + "maxReplicas": { + "type": "integer" + }, + "minReplicas": { + "type": "integer" + }, + "targetCPUUtilizationPercentage": { + "type": "integer" + }, + "targetMemoryUtilizationPercentage": { + "type": "integer" + } + }, + "type": "object" + }, + "commonLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "configmap": { + "properties": { + "create": { + "type": "boolean" + } + }, + "type": "object" + }, + "console": { + "properties": { + "config": { + "type": "object" + }, + "roleBindings": { + "oneOf": [ + { + "items": { + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "roles": { + "oneOf": [ + { + "items": { + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "deployment": { + "properties": { + "command": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "create": { + "type": "boolean" + }, + "extraArgs": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "enterprise": { + "properties": { + "licenseSecretRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "extraContainers": { + "oneOf": [ + { + "items": { + "properties": { + "args": { + "items": { + "type": "string" + }, + "type": "array" + }, + "command": { + "items": { + "type": "string" + }, + "type": "array" + }, + "env": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "envFrom": { + "items": { + "properties": { + "configMapRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "prefix": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "image": { + "type": "string" + }, + "imagePullPolicy": { + "type": "string" + }, + "lifecycle": { + "properties": { + "postStart": { + "properties": { + "exec": { + "properties": { + "command": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "sleep": { + "properties": { + "seconds": { + "type": "integer" + } + }, + "type": "object" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "preStop": { + "properties": { + "exec": { + "properties": { + "command": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "sleep": { + "properties": { + "seconds": { + "type": "integer" + } + }, + "type": "object" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "livenessProbe": { + "properties": { + "exec": { + "properties": { + "command": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "failureThreshold": { + "type": "integer" + }, + "grpc": { + "properties": { + "port": { + "type": "integer" + }, + "service": { + "type": "string" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "ports": { + "items": { + "properties": { + "containerPort": { + "type": "integer" + }, + "hostIP": { + "type": "string" + }, + "hostPort": { + "type": "integer" + }, + "name": { + "type": "string" + }, + "protocol": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "readinessProbe": { + "properties": { + "exec": { + "properties": { + "command": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "failureThreshold": { + "type": "integer" + }, + "grpc": { + "properties": { + "port": { + "type": "integer" + }, + "service": { + "type": "string" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "type": "object" + }, + "resizePolicy": { + "items": { + "properties": { + "resourceName": { + "type": "string" + }, + "restartPolicy": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "resources": { + "properties": { + "claims": { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "restartPolicy": { + "type": "string" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "items": { + "type": "string" + }, + "type": "array" + }, + "drop": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "startupProbe": { + "properties": { + "exec": { + "properties": { + "command": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "failureThreshold": { + "type": "integer" + }, + "grpc": { + "properties": { + "port": { + "type": "integer" + }, + "service": { + "type": "string" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "type": "object" + }, + "stdin": { + "type": "boolean" + }, + "stdinOnce": { + "type": "boolean" + }, + "terminationMessagePath": { + "type": "string" + }, + "terminationMessagePolicy": { + "type": "string" + }, + "tty": { + "type": "boolean" + }, + "volumeDevices": { + "items": { + "properties": { + "devicePath": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "volumeMounts": { + "items": { + "properties": { + "mountPath": { + "type": "string" + }, + "mountPropagation": { + "type": "string" + }, + "name": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "subPath": { + "type": "string" + }, + "subPathExpr": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "workingDir": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "extraEnv": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "extraEnvFrom": { + "oneOf": [ + { + "items": { + "properties": { + "configMapRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "prefix": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "extraVolumeMounts": { + "oneOf": [ + { + "items": { + "properties": { + "mountPath": { + "type": "string" + }, + "mountPropagation": { + "type": "string" + }, + "name": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "subPath": { + "type": "string" + }, + "subPathExpr": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "extraVolumes": { + "oneOf": [ + { + "items": { + "properties": { + "awsElasticBlockStore": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "azureDisk": { + "properties": { + "cachingMode": { + "type": "string" + }, + "diskName": { + "type": "string" + }, + "diskURI": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "azureFile": { + "properties": { + "readOnly": { + "type": "boolean" + }, + "secretName": { + "type": "string" + }, + "shareName": { + "type": "string" + } + }, + "type": "object" + }, + "cephfs": { + "properties": { + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretFile": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "cinder": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "csi": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "nodePublishSecretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "volumeAttributes": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "emptyDir": { + "properties": { + "medium": { + "type": "string" + }, + "sizeLimit": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "type": "object" + }, + "ephemeral": { + "properties": { + "volumeClaimTemplate": { + "properties": { + "metadata": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "creationTimestamp": { + "properties": {}, + "type": "object" + }, + "deletionGracePeriodSeconds": { + "type": "integer" + }, + "deletionTimestamp": { + "properties": {}, + "type": "object" + }, + "finalizers": { + "items": { + "type": "string" + }, + "type": "array" + }, + "generateName": { + "type": "string" + }, + "generation": { + "type": "integer" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "managedFields": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldsType": { + "type": "string" + }, + "fieldsV1": { + "properties": {}, + "type": "object" + }, + "manager": { + "type": "string" + }, + "operation": { + "type": "string" + }, + "subresource": { + "type": "string" + }, + "time": { + "properties": {}, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + }, + "ownerReferences": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "blockOwnerDeletion": { + "type": "boolean" + }, + "controller": { + "type": "boolean" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "resourceVersion": { + "type": "string" + }, + "selfLink": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "spec": { + "properties": { + "accessModes": { + "items": { + "type": "string" + }, + "type": "array" + }, + "dataSource": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "dataSourceRef": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + } + }, + "type": "object" + }, + "resources": { + "properties": { + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "selector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "storageClassName": { + "type": "string" + }, + "volumeAttributesClassName": { + "type": "string" + }, + "volumeMode": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "fc": { + "properties": { + "fsType": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "targetWWNs": { + "items": { + "type": "string" + }, + "type": "array" + }, + "wwids": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "flexVolume": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "options": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "flocker": { + "properties": { + "datasetName": { + "type": "string" + }, + "datasetUUID": { + "type": "string" + } + }, + "type": "object" + }, + "gcePersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "pdName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "gitRepo": { + "properties": { + "directory": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "revision": { + "type": "string" + } + }, + "type": "object" + }, + "glusterfs": { + "properties": { + "endpoints": { + "type": "string" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "hostPath": { + "properties": { + "path": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "iscsi": { + "properties": { + "chapAuthDiscovery": { + "type": "boolean" + }, + "chapAuthSession": { + "type": "boolean" + }, + "fsType": { + "type": "string" + }, + "initiatorName": { + "type": "string" + }, + "iqn": { + "type": "string" + }, + "iscsiInterface": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "portals": { + "items": { + "type": "string" + }, + "type": "array" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "targetPortal": { + "type": "string" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "nfs": { + "properties": { + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "server": { + "type": "string" + } + }, + "type": "object" + }, + "persistentVolumeClaim": { + "properties": { + "claimName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "photonPersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "pdID": { + "type": "string" + } + }, + "type": "object" + }, + "portworxVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "projected": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "sources": { + "items": { + "properties": { + "clusterTrustBundle": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + }, + "path": { + "type": "string" + }, + "signerName": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "serviceAccountToken": { + "properties": { + "audience": { + "type": "string" + }, + "expirationSeconds": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "quobyte": { + "properties": { + "group": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "registry": { + "type": "string" + }, + "tenant": { + "type": "string" + }, + "user": { + "type": "string" + }, + "volume": { + "type": "string" + } + }, + "type": "object" + }, + "rbd": { + "properties": { + "fsType": { + "type": "string" + }, + "image": { + "type": "string" + }, + "keyring": { + "type": "string" + }, + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "pool": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "scaleIO": { + "properties": { + "fsType": { + "type": "string" + }, + "gateway": { + "type": "string" + }, + "protectionDomain": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "sslEnabled": { + "type": "boolean" + }, + "storageMode": { + "type": "string" + }, + "storagePool": { + "type": "string" + }, + "system": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "optional": { + "type": "boolean" + }, + "secretName": { + "type": "string" + } + }, + "type": "object" + }, + "storageos": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeName": { + "type": "string" + }, + "volumeNamespace": { + "type": "string" + } + }, + "type": "object" + }, + "vsphereVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "storagePolicyID": { + "type": "string" + }, + "storagePolicyName": { + "type": "string" + }, + "volumePath": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "fullnameOverride": { + "type": "string" + }, + "image": { + "properties": { + "pullPolicy": { + "type": "string" + }, + "registry": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "tag": { + "type": "string" + } + }, + "type": "object" + }, + "imagePullSecrets": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "ingress": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "className": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "hosts": { + "oneOf": [ + { + "items": { + "properties": { + "host": { + "type": "string" + }, + "paths": { + "items": { + "properties": { + "path": { + "type": "string" + }, + "pathType": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "tls": { + "oneOf": [ + { + "items": { + "properties": { + "hosts": { + "items": { + "type": "string" + }, + "type": "array" + }, + "secretName": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "initContainers": { + "properties": { + "extraInitContainers": { + "type": "string" + } + }, + "type": "object" + }, + "livenessProbe": { + "properties": { + "exec": { + "properties": { + "command": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "failureThreshold": { + "type": "integer" + }, + "grpc": { + "properties": { + "port": { + "type": "integer" + }, + "service": { + "type": "string" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "type": "object" + }, + "nameOverride": { + "type": "string" + }, + "nodeSelector": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podAnnotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podSecurityContext": { + "properties": { + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "supplementalGroups": { + "oneOf": [ + { + "items": { + "type": "integer" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "sysctls": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "priorityClassName": { + "type": "string" + }, + "readinessProbe": { + "properties": { + "exec": { + "properties": { + "command": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "failureThreshold": { + "type": "integer" + }, + "grpc": { + "properties": { + "port": { + "type": "integer" + }, + "service": { + "type": "string" + } + }, + "type": "object" + }, + "httpGet": { + "properties": { + "host": { + "type": "string" + }, + "httpHeaders": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "path": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "scheme": { + "type": "string" + } + }, + "type": "object" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "tcpSocket": { + "properties": { + "host": { + "type": "string" + }, + "port": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "type": "object" + }, + "replicaCount": { + "type": "integer" + }, + "resources": { + "properties": { + "claims": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "create": { + "type": "boolean" + }, + "enterprise": { + "properties": { + "license": { + "type": "string" + } + }, + "type": "object" + }, + "kafka": { + "properties": { + "awsMskIamSecretKey": { + "type": "string" + }, + "protobufGitBasicAuthPassword": { + "type": "string" + }, + "saslPassword": { + "type": "string" + }, + "schemaRegistryPassword": { + "type": "string" + }, + "schemaRegistryTlsCa": { + "type": "string" + }, + "schemaRegistryTlsCert": { + "type": "string" + }, + "schemaRegistryTlsKey": { + "type": "string" + }, + "tlsCa": { + "type": "string" + }, + "tlsCert": { + "type": "string" + }, + "tlsKey": { + "type": "string" + }, + "tlsPassphrase": { + "type": "string" + } + }, + "type": "object" + }, + "login": { + "properties": { + "github": { + "properties": { + "clientSecret": { + "type": "string" + }, + "personalAccessToken": { + "type": "string" + } + }, + "type": "object" + }, + "google": { + "properties": { + "clientSecret": { + "type": "string" + }, + "groupsServiceAccount": { + "type": "string" + } + }, + "type": "object" + }, + "jwtSecret": { + "type": "string" + }, + "oidc": { + "properties": { + "clientSecret": { + "type": "string" + } + }, + "type": "object" + }, + "okta": { + "properties": { + "clientSecret": { + "type": "string" + }, + "directoryApiToken": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "redpanda": { + "properties": { + "adminApi": { + "properties": { + "password": { + "type": "string" + }, + "tlsCa": { + "type": "string" + }, + "tlsCert": { + "type": "string" + }, + "tlsKey": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "secretMounts": { + "oneOf": [ + { + "items": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "name": { + "type": "string" + }, + "path": { + "type": "string" + }, + "secretName": { + "type": "string" + }, + "subPath": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "drop": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "service": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "nodePort": { + "type": "integer" + }, + "port": { + "type": "integer" + }, + "targetPort": { + "type": "integer" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "serviceAccount": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "automountServiceAccountToken": { + "type": "boolean" + }, + "create": { + "type": "boolean" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "strategy": { + "properties": { + "rollingUpdate": { + "properties": { + "maxSurge": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + }, + "maxUnavailable": { + "oneOf": [ + { + "type": "string" + }, + { + "type": "number" + } + ] + } + }, + "type": "object" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "tests": { + "properties": { + "enabled": { + "type": "boolean" + } + }, + "type": "object" + }, + "tolerations": { + "oneOf": [ + { + "items": { + "properties": { + "effect": { + "type": "string" + }, + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "tolerationSeconds": { + "type": "integer" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "topologySpreadConstraints": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "maxSkew": { + "type": "integer" + }, + "minDomains": { + "type": "integer" + }, + "nodeAffinityPolicy": { + "type": "string" + }, + "nodeTaintsPolicy": { + "type": "string" + }, + "topologyKey": { + "type": "string" + }, + "whenUnsatisfiable": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "enterprise": { + "properties": { + "license": { + "type": "string" + }, + "licenseSecretRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "external": { + "properties": { + "addresses": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "domain": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "externalDns": { + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "type": "object" + }, + "prefixTemplate": { + "type": "string" + }, + "service": { + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "type": "object" + }, + "sourceRanges": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "type": { + "pattern": "^(LoadBalancer|NodePort)$", + "type": "string" + } + }, + "required": [ + "enabled" + ], + "type": "object" + }, + "force": { + "type": "boolean" + }, + "fullnameOverride": { + "type": "string" + }, + "image": { + "description": "Values used to define the container image to be used for Redpanda", + "properties": { + "pullPolicy": { + "description": "The Kubernetes Pod image pull policy.", + "pattern": "^(Always|Never|IfNotPresent)$", + "type": "string" + }, + "repository": { + "default": "docker.redpanda.com/redpandadata/redpanda", + "description": "container image repository", + "type": "string" + }, + "tag": { + "default": "Chart.appVersion", + "description": "The container image tag. Use the Redpanda release version. Must be a valid semver prefixed with a 'v'.", + "pattern": "^v(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$|^$", + "type": "string" + } + }, + "required": [ + "repository", + "pullPolicy" + ], + "type": "object" + }, + "imagePullSecrets": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "license_key": { + "deprecated": true, + "pattern": "^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?\\.(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$|^$", + "type": "string" + }, + "license_secret_ref": { + "deprecated": true, + "properties": { + "secret_key": { + "type": "string" + }, + "secret_name": { + "type": "string" + } + }, + "type": "object" + }, + "listeners": { + "properties": { + "admin": { + "properties": { + "appProtocol": { + "type": "string" + }, + "external": { + "minProperties": 1, + "patternProperties": { + "^[A-Za-z_][A-Za-z0-9_]*$": { + "properties": { + "advertisedPorts": { + "items": { + "type": "integer" + }, + "minItems": 1, + "type": "array" + }, + "enabled": { + "type": "boolean" + }, + "nodePort": { + "type": "integer" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "port" + ], + "type": "object" + } + }, + "type": "object" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "cert", + "requireClientAuth" + ], + "type": "object" + } + }, + "required": [ + "port", + "tls" + ], + "type": "object" + }, + "http": { + "properties": { + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "none", + "http_basic" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "enabled": { + "type": "boolean" + }, + "external": { + "minProperties": 1, + "patternProperties": { + "^[A-Za-z_][A-Za-z0-9_]*$": { + "properties": { + "advertisedPorts": { + "items": { + "type": "integer" + }, + "minItems": 1, + "type": "array" + }, + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "none", + "http_basic" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "enabled": { + "type": "boolean" + }, + "nodePort": { + "type": "integer" + }, + "port": { + "type": "integer" + }, + "prefixTemplate": { + "type": "string" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "port" + ], + "type": "object" + } + }, + "type": "object" + }, + "kafkaEndpoint": { + "pattern": "^[A-Za-z_-][A-Za-z0-9_-]*$", + "type": "string" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "cert", + "requireClientAuth" + ], + "type": "object" + } + }, + "required": [ + "enabled", + "tls", + "kafkaEndpoint", + "port" + ], + "type": "object" + }, + "kafka": { + "properties": { + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "sasl", + "none", + "mtls_identity" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "external": { + "minProperties": 1, + "patternProperties": { + "^[A-Za-z_][A-Za-z0-9_]*$": { + "properties": { + "advertisedPorts": { + "items": { + "type": "integer" + }, + "minItems": 1, + "type": "array" + }, + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "sasl", + "none", + "mtls_identity" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "enabled": { + "type": "boolean" + }, + "nodePort": { + "type": "integer" + }, + "port": { + "type": "integer" + }, + "prefixTemplate": { + "type": "string" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "port" + ], + "type": "object" + } + }, + "type": "object" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "cert", + "requireClientAuth" + ], + "type": "object" + } + }, + "required": [ + "tls", + "port" + ], + "type": "object" + }, + "rpc": { + "properties": { + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "cert", + "requireClientAuth" + ], + "type": "object" + } + }, + "required": [ + "port", + "tls" + ], + "type": "object" + }, + "schemaRegistry": { + "properties": { + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "none", + "http_basic" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "enabled": { + "type": "boolean" + }, + "external": { + "minProperties": 1, + "patternProperties": { + "^[A-Za-z_][A-Za-z0-9_]*$": { + "properties": { + "advertisedPorts": { + "items": { + "type": "integer" + }, + "minItems": 1, + "type": "array" + }, + "authenticationMethod": { + "oneOf": [ + { + "enum": [ + "none", + "http_basic" + ], + "type": "string" + }, + { + "type": "null" + } + ] + }, + "enabled": { + "type": "boolean" + }, + "nodePort": { + "type": "integer" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "kafkaEndpoint": { + "pattern": "^[A-Za-z_-][A-Za-z0-9_-]*$", + "type": "string" + }, + "port": { + "type": "integer" + }, + "tls": { + "properties": { + "cert": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "requireClientAuth": { + "type": "boolean" + }, + "trustStore": { + "maxProperties": 1, + "minProperties": 1, + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "required": [ + "cert", + "requireClientAuth" + ], + "type": "object" + } + }, + "required": [ + "enabled", + "kafkaEndpoint", + "port", + "tls" + ], + "type": "object" + } + }, + "required": [ + "admin", + "http", + "kafka", + "schemaRegistry", + "rpc" + ], + "type": "object" + }, + "logging": { + "properties": { + "logLevel": { + "pattern": "^(error|warn|info|debug|trace)$", + "type": "string" + }, + "usageStats": { + "properties": { + "clusterId": { + "type": "string" + }, + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "type": "object" + } + }, + "required": [ + "logLevel", + "usageStats" + ], + "type": "object" + }, + "monitoring": { + "properties": { + "enableHttp2": { + "type": "boolean" + }, + "enabled": { + "type": "boolean" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "scrapeInterval": { + "type": "string" + }, + "tlsConfig": { + "properties": { + "ca": { + "properties": { + "configMap": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "caFile": { + "type": "string" + }, + "cert": { + "properties": { + "configMap": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "certFile": { + "type": "string" + }, + "insecureSkipVerify": { + "type": "boolean" + }, + "keyFile": { + "type": "string" + }, + "keySecret": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "serverName": { + "type": "string" + } + }, + "type": "object" + } + }, + "required": [ + "enabled", + "scrapeInterval" + ], + "type": "object" + }, + "nameOverride": { + "type": "string" + }, + "nodeSelector": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "post_install_job": { + "properties": { + "affinity": { + "properties": { + "nodeAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "preference": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "properties": { + "nodeSelectorTerms": { + "oneOf": [ + { + "items": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "podAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "podAntiAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podTemplate": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "spec": { + "properties": { + "automountServiceAccountToken": { + "type": "boolean" + }, + "containers": { + "oneOf": [ + { + "items": { + "properties": { + "env": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "enum": [ + "redpanda", + "post-install", + "post-upgrade", + "redpanda-controllers" + ], + "type": "string" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "items": { + "type": "string" + }, + "type": "array" + }, + "drop": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumeMounts": { + "items": { + "properties": { + "mountPath": { + "type": "string" + }, + "mountPropagation": { + "type": "string" + }, + "name": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "subPath": { + "type": "string" + }, + "subPathExpr": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "required": [ + "name", + "env" + ], + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "securityContext": { + "properties": { + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "supplementalGroups": { + "oneOf": [ + { + "items": { + "type": "integer" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "sysctls": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumes": { + "oneOf": [ + { + "items": { + "properties": { + "awsElasticBlockStore": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "azureDisk": { + "properties": { + "cachingMode": { + "type": "string" + }, + "diskName": { + "type": "string" + }, + "diskURI": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "azureFile": { + "properties": { + "readOnly": { + "type": "boolean" + }, + "secretName": { + "type": "string" + }, + "shareName": { + "type": "string" + } + }, + "type": "object" + }, + "cephfs": { + "properties": { + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretFile": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "cinder": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "csi": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "nodePublishSecretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "volumeAttributes": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "emptyDir": { + "properties": { + "medium": { + "type": "string" + }, + "sizeLimit": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "type": "object" + }, + "ephemeral": { + "properties": { + "volumeClaimTemplate": { + "properties": { + "metadata": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "creationTimestamp": { + "properties": {}, + "type": "object" + }, + "deletionGracePeriodSeconds": { + "type": "integer" + }, + "deletionTimestamp": { + "properties": {}, + "type": "object" + }, + "finalizers": { + "items": { + "type": "string" + }, + "type": "array" + }, + "generateName": { + "type": "string" + }, + "generation": { + "type": "integer" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "managedFields": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldsType": { + "type": "string" + }, + "fieldsV1": { + "properties": {}, + "type": "object" + }, + "manager": { + "type": "string" + }, + "operation": { + "type": "string" + }, + "subresource": { + "type": "string" + }, + "time": { + "properties": {}, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + }, + "ownerReferences": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "blockOwnerDeletion": { + "type": "boolean" + }, + "controller": { + "type": "boolean" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "resourceVersion": { + "type": "string" + }, + "selfLink": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "spec": { + "properties": { + "accessModes": { + "items": { + "type": "string" + }, + "type": "array" + }, + "dataSource": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "dataSourceRef": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + } + }, + "type": "object" + }, + "resources": { + "properties": { + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "selector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "storageClassName": { + "type": "string" + }, + "volumeAttributesClassName": { + "type": "string" + }, + "volumeMode": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "fc": { + "properties": { + "fsType": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "targetWWNs": { + "items": { + "type": "string" + }, + "type": "array" + }, + "wwids": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "flexVolume": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "options": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "flocker": { + "properties": { + "datasetName": { + "type": "string" + }, + "datasetUUID": { + "type": "string" + } + }, + "type": "object" + }, + "gcePersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "pdName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "gitRepo": { + "properties": { + "directory": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "revision": { + "type": "string" + } + }, + "type": "object" + }, + "glusterfs": { + "properties": { + "endpoints": { + "type": "string" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "hostPath": { + "properties": { + "path": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "iscsi": { + "properties": { + "chapAuthDiscovery": { + "type": "boolean" + }, + "chapAuthSession": { + "type": "boolean" + }, + "fsType": { + "type": "string" + }, + "initiatorName": { + "type": "string" + }, + "iqn": { + "type": "string" + }, + "iscsiInterface": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "portals": { + "items": { + "type": "string" + }, + "type": "array" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "targetPortal": { + "type": "string" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "nfs": { + "properties": { + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "server": { + "type": "string" + } + }, + "type": "object" + }, + "persistentVolumeClaim": { + "properties": { + "claimName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "photonPersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "pdID": { + "type": "string" + } + }, + "type": "object" + }, + "portworxVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "projected": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "sources": { + "items": { + "properties": { + "clusterTrustBundle": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + }, + "path": { + "type": "string" + }, + "signerName": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "serviceAccountToken": { + "properties": { + "audience": { + "type": "string" + }, + "expirationSeconds": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "quobyte": { + "properties": { + "group": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "registry": { + "type": "string" + }, + "tenant": { + "type": "string" + }, + "user": { + "type": "string" + }, + "volume": { + "type": "string" + } + }, + "type": "object" + }, + "rbd": { + "properties": { + "fsType": { + "type": "string" + }, + "image": { + "type": "string" + }, + "keyring": { + "type": "string" + }, + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "pool": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "scaleIO": { + "properties": { + "fsType": { + "type": "string" + }, + "gateway": { + "type": "string" + }, + "protectionDomain": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "sslEnabled": { + "type": "boolean" + }, + "storageMode": { + "type": "string" + }, + "storagePool": { + "type": "string" + }, + "system": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "optional": { + "type": "boolean" + }, + "secretName": { + "type": "string" + } + }, + "type": "object" + }, + "storageos": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeName": { + "type": "string" + }, + "volumeNamespace": { + "type": "string" + } + }, + "type": "object" + }, + "vsphereVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "storagePolicyID": { + "type": "string" + }, + "storagePolicyName": { + "type": "string" + }, + "volumePath": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "required": [ + "containers" + ], + "type": "object" + } + }, + "required": [ + "labels", + "annotations", + "spec" + ], + "type": "object" + }, + "resources": { + "properties": { + "claims": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "drop": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "post_upgrade_job": { + "properties": { + "affinity": { + "properties": { + "nodeAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "preference": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "properties": { + "nodeSelectorTerms": { + "oneOf": [ + { + "items": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchFields": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "podAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "podAntiAffinity": { + "properties": { + "preferredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "podAffinityTerm": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "weight": { + "type": "integer" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "requiredDuringSchedulingIgnoredDuringExecution": { + "oneOf": [ + { + "items": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "matchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "mismatchLabelKeys": { + "items": { + "type": "string" + }, + "type": "array" + }, + "namespaceSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "namespaces": { + "items": { + "type": "string" + }, + "type": "array" + }, + "topologyKey": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + } + }, + "type": "object" + }, + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "backoffLimit": { + "type": "integer" + }, + "enabled": { + "type": "boolean" + }, + "extraEnv": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "extraEnvFrom": { + "oneOf": [ + { + "items": { + "properties": { + "configMapRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "prefix": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podTemplate": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "spec": { + "properties": { + "automountServiceAccountToken": { + "type": "boolean" + }, + "containers": { + "oneOf": [ + { + "items": { + "properties": { + "env": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "enum": [ + "redpanda", + "post-install", + "post-upgrade", + "redpanda-controllers" + ], + "type": "string" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "items": { + "type": "string" + }, + "type": "array" + }, + "drop": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumeMounts": { + "items": { + "properties": { + "mountPath": { + "type": "string" + }, + "mountPropagation": { + "type": "string" + }, + "name": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "subPath": { + "type": "string" + }, + "subPathExpr": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "required": [ + "name", + "env" + ], + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "securityContext": { + "properties": { + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "supplementalGroups": { + "oneOf": [ + { + "items": { + "type": "integer" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "sysctls": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumes": { + "oneOf": [ + { + "items": { + "properties": { + "awsElasticBlockStore": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "azureDisk": { + "properties": { + "cachingMode": { + "type": "string" + }, + "diskName": { + "type": "string" + }, + "diskURI": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "azureFile": { + "properties": { + "readOnly": { + "type": "boolean" + }, + "secretName": { + "type": "string" + }, + "shareName": { + "type": "string" + } + }, + "type": "object" + }, + "cephfs": { + "properties": { + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretFile": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "cinder": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "csi": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "nodePublishSecretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "volumeAttributes": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "emptyDir": { + "properties": { + "medium": { + "type": "string" + }, + "sizeLimit": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "type": "object" + }, + "ephemeral": { + "properties": { + "volumeClaimTemplate": { + "properties": { + "metadata": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "creationTimestamp": { + "properties": {}, + "type": "object" + }, + "deletionGracePeriodSeconds": { + "type": "integer" + }, + "deletionTimestamp": { + "properties": {}, + "type": "object" + }, + "finalizers": { + "items": { + "type": "string" + }, + "type": "array" + }, + "generateName": { + "type": "string" + }, + "generation": { + "type": "integer" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "managedFields": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldsType": { + "type": "string" + }, + "fieldsV1": { + "properties": {}, + "type": "object" + }, + "manager": { + "type": "string" + }, + "operation": { + "type": "string" + }, + "subresource": { + "type": "string" + }, + "time": { + "properties": {}, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + }, + "ownerReferences": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "blockOwnerDeletion": { + "type": "boolean" + }, + "controller": { + "type": "boolean" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "resourceVersion": { + "type": "string" + }, + "selfLink": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "spec": { + "properties": { + "accessModes": { + "items": { + "type": "string" + }, + "type": "array" + }, + "dataSource": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "dataSourceRef": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + } + }, + "type": "object" + }, + "resources": { + "properties": { + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "selector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "storageClassName": { + "type": "string" + }, + "volumeAttributesClassName": { + "type": "string" + }, + "volumeMode": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "fc": { + "properties": { + "fsType": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "targetWWNs": { + "items": { + "type": "string" + }, + "type": "array" + }, + "wwids": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "flexVolume": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "options": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "flocker": { + "properties": { + "datasetName": { + "type": "string" + }, + "datasetUUID": { + "type": "string" + } + }, + "type": "object" + }, + "gcePersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "pdName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "gitRepo": { + "properties": { + "directory": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "revision": { + "type": "string" + } + }, + "type": "object" + }, + "glusterfs": { + "properties": { + "endpoints": { + "type": "string" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "hostPath": { + "properties": { + "path": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "iscsi": { + "properties": { + "chapAuthDiscovery": { + "type": "boolean" + }, + "chapAuthSession": { + "type": "boolean" + }, + "fsType": { + "type": "string" + }, + "initiatorName": { + "type": "string" + }, + "iqn": { + "type": "string" + }, + "iscsiInterface": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "portals": { + "items": { + "type": "string" + }, + "type": "array" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "targetPortal": { + "type": "string" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "nfs": { + "properties": { + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "server": { + "type": "string" + } + }, + "type": "object" + }, + "persistentVolumeClaim": { + "properties": { + "claimName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "photonPersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "pdID": { + "type": "string" + } + }, + "type": "object" + }, + "portworxVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "projected": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "sources": { + "items": { + "properties": { + "clusterTrustBundle": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + }, + "path": { + "type": "string" + }, + "signerName": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "serviceAccountToken": { + "properties": { + "audience": { + "type": "string" + }, + "expirationSeconds": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "quobyte": { + "properties": { + "group": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "registry": { + "type": "string" + }, + "tenant": { + "type": "string" + }, + "user": { + "type": "string" + }, + "volume": { + "type": "string" + } + }, + "type": "object" + }, + "rbd": { + "properties": { + "fsType": { + "type": "string" + }, + "image": { + "type": "string" + }, + "keyring": { + "type": "string" + }, + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "pool": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "scaleIO": { + "properties": { + "fsType": { + "type": "string" + }, + "gateway": { + "type": "string" + }, + "protectionDomain": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "sslEnabled": { + "type": "boolean" + }, + "storageMode": { + "type": "string" + }, + "storagePool": { + "type": "string" + }, + "system": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "optional": { + "type": "boolean" + }, + "secretName": { + "type": "string" + } + }, + "type": "object" + }, + "storageos": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeName": { + "type": "string" + }, + "volumeNamespace": { + "type": "string" + } + }, + "type": "object" + }, + "vsphereVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "storagePolicyID": { + "type": "string" + }, + "storagePolicyName": { + "type": "string" + }, + "volumePath": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "required": [ + "containers" + ], + "type": "object" + } + }, + "required": [ + "labels", + "annotations", + "spec" + ], + "type": "object" + }, + "resources": { + "properties": { + "claims": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "drop": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "rackAwareness": { + "properties": { + "enabled": { + "type": "boolean" + }, + "nodeAnnotation": { + "type": "string" + } + }, + "required": [ + "enabled", + "nodeAnnotation" + ], + "type": "object" + }, + "rbac": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled", + "annotations" + ], + "type": "object" + }, + "resources": { + "properties": { + "cpu": { + "properties": { + "cores": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "overprovisioned": { + "type": "boolean" + } + }, + "required": [ + "cores" + ], + "type": "object" + }, + "memory": { + "properties": { + "container": { + "properties": { + "max": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "min": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "required": [ + "max" + ], + "type": "object" + }, + "enable_memory_locking": { + "type": "boolean" + }, + "redpanda": { + "properties": { + "memory": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "reserveMemory": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "type": "object" + } + }, + "required": [ + "container" + ], + "type": "object" + } + }, + "required": [ + "cpu", + "memory" + ], + "type": "object" + }, + "service": { + "properties": { + "internal": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "serviceAccount": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "automountServiceAccountToken": { + "type": "boolean" + }, + "create": { + "type": "boolean" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "type": "object" + }, + "statefulset": { + "properties": { + "additionalRedpandaCmdFlags": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "additionalSelectorLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "budget": { + "properties": { + "maxUnavailable": { + "type": "integer" + } + }, + "required": [ + "maxUnavailable" + ], + "type": "object" + }, + "extraVolumeMounts": { + "type": "string" + }, + "extraVolumes": { + "type": "string" + }, + "initContainerImage": { + "properties": { + "repository": { + "type": "string" + }, + "tag": { + "type": "string" + } + }, + "type": "object" + }, + "initContainers": { + "properties": { + "configurator": { + "properties": { + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + } + }, + "type": "object" + }, + "extraInitContainers": { + "type": "string" + }, + "fsValidator": { + "properties": { + "enabled": { + "type": "boolean" + }, + "expectedFS": { + "type": "string" + }, + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + } + }, + "type": "object" + }, + "setDataDirOwnership": { + "properties": { + "enabled": { + "type": "boolean" + }, + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + } + }, + "type": "object" + }, + "setTieredStorageCacheDirOwnership": { + "properties": { + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + } + }, + "type": "object" + }, + "tuning": { + "properties": { + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "livenessProbe": { + "properties": { + "failureThreshold": { + "type": "integer" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + } + }, + "required": [ + "initialDelaySeconds", + "failureThreshold", + "periodSeconds" + ], + "type": "object" + }, + "nodeAffinity": { + "type": "object" + }, + "nodeSelector": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "podAffinity": { + "type": "object" + }, + "podAntiAffinity": { + "properties": { + "custom": { + "type": "object" + }, + "topologyKey": { + "type": "string" + }, + "type": { + "pattern": "^(hard|soft|custom)$", + "type": "string" + }, + "weight": { + "type": "integer" + } + }, + "required": [ + "topologyKey", + "type", + "weight" + ], + "type": "object" + }, + "podSecurityContext": { + "deprecated": true, + "properties": { + "allowPriviledgeEscalation": { + "type": "boolean" + }, + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + } + }, + "type": "object" + }, + "podTemplate": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "spec": { + "properties": { + "automountServiceAccountToken": { + "type": "boolean" + }, + "containers": { + "oneOf": [ + { + "items": { + "properties": { + "env": { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + }, + "valueFrom": { + "properties": { + "configMapKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + }, + "secretKeyRef": { + "properties": { + "key": { + "type": "string" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "enum": [ + "redpanda", + "post-install", + "post-upgrade", + "redpanda-controllers" + ], + "type": "string" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "items": { + "type": "string" + }, + "type": "array" + }, + "drop": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumeMounts": { + "items": { + "properties": { + "mountPath": { + "type": "string" + }, + "mountPropagation": { + "type": "string" + }, + "name": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "subPath": { + "type": "string" + }, + "subPathExpr": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "required": [ + "name", + "env" + ], + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "securityContext": { + "properties": { + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "supplementalGroups": { + "oneOf": [ + { + "items": { + "type": "integer" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "sysctls": { + "oneOf": [ + { + "items": { + "properties": { + "name": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "volumes": { + "oneOf": [ + { + "items": { + "properties": { + "awsElasticBlockStore": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "azureDisk": { + "properties": { + "cachingMode": { + "type": "string" + }, + "diskName": { + "type": "string" + }, + "diskURI": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "azureFile": { + "properties": { + "readOnly": { + "type": "boolean" + }, + "secretName": { + "type": "string" + }, + "shareName": { + "type": "string" + } + }, + "type": "object" + }, + "cephfs": { + "properties": { + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretFile": { + "type": "string" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "cinder": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "csi": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "nodePublishSecretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "volumeAttributes": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "emptyDir": { + "properties": { + "medium": { + "type": "string" + }, + "sizeLimit": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + } + }, + "type": "object" + }, + "ephemeral": { + "properties": { + "volumeClaimTemplate": { + "properties": { + "metadata": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "creationTimestamp": { + "properties": {}, + "type": "object" + }, + "deletionGracePeriodSeconds": { + "type": "integer" + }, + "deletionTimestamp": { + "properties": {}, + "type": "object" + }, + "finalizers": { + "items": { + "type": "string" + }, + "type": "array" + }, + "generateName": { + "type": "string" + }, + "generation": { + "type": "integer" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "managedFields": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldsType": { + "type": "string" + }, + "fieldsV1": { + "properties": {}, + "type": "object" + }, + "manager": { + "type": "string" + }, + "operation": { + "type": "string" + }, + "subresource": { + "type": "string" + }, + "time": { + "properties": {}, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + }, + "ownerReferences": { + "items": { + "properties": { + "apiVersion": { + "type": "string" + }, + "blockOwnerDeletion": { + "type": "boolean" + }, + "controller": { + "type": "boolean" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "resourceVersion": { + "type": "string" + }, + "selfLink": { + "type": "string" + }, + "uid": { + "type": "string" + } + }, + "type": "object" + }, + "spec": { + "properties": { + "accessModes": { + "items": { + "type": "string" + }, + "type": "array" + }, + "dataSource": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "dataSourceRef": { + "properties": { + "apiGroup": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + }, + "namespace": { + "type": "string" + } + }, + "type": "object" + }, + "resources": { + "properties": { + "limits": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + }, + "requests": { + "additionalProperties": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "type": "object" + } + }, + "type": "object" + }, + "selector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "storageClassName": { + "type": "string" + }, + "volumeAttributesClassName": { + "type": "string" + }, + "volumeMode": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "fc": { + "properties": { + "fsType": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "readOnly": { + "type": "boolean" + }, + "targetWWNs": { + "items": { + "type": "string" + }, + "type": "array" + }, + "wwids": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "flexVolume": { + "properties": { + "driver": { + "type": "string" + }, + "fsType": { + "type": "string" + }, + "options": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "flocker": { + "properties": { + "datasetName": { + "type": "string" + }, + "datasetUUID": { + "type": "string" + } + }, + "type": "object" + }, + "gcePersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "partition": { + "type": "integer" + }, + "pdName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "gitRepo": { + "properties": { + "directory": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "revision": { + "type": "string" + } + }, + "type": "object" + }, + "glusterfs": { + "properties": { + "endpoints": { + "type": "string" + }, + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "hostPath": { + "properties": { + "path": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "iscsi": { + "properties": { + "chapAuthDiscovery": { + "type": "boolean" + }, + "chapAuthSession": { + "type": "boolean" + }, + "fsType": { + "type": "string" + }, + "initiatorName": { + "type": "string" + }, + "iqn": { + "type": "string" + }, + "iscsiInterface": { + "type": "string" + }, + "lun": { + "type": "integer" + }, + "portals": { + "items": { + "type": "string" + }, + "type": "array" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "targetPortal": { + "type": "string" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "nfs": { + "properties": { + "path": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "server": { + "type": "string" + } + }, + "type": "object" + }, + "persistentVolumeClaim": { + "properties": { + "claimName": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + } + }, + "type": "object" + }, + "photonPersistentDisk": { + "properties": { + "fsType": { + "type": "string" + }, + "pdID": { + "type": "string" + } + }, + "type": "object" + }, + "portworxVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "volumeID": { + "type": "string" + } + }, + "type": "object" + }, + "projected": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "sources": { + "items": { + "properties": { + "clusterTrustBundle": { + "properties": { + "labelSelector": { + "properties": { + "matchExpressions": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "array" + }, + "matchLabels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + } + }, + "type": "object" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + }, + "path": { + "type": "string" + }, + "signerName": { + "type": "string" + } + }, + "type": "object" + }, + "configMap": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "downwardAPI": { + "properties": { + "items": { + "items": { + "properties": { + "fieldRef": { + "properties": { + "apiVersion": { + "type": "string" + }, + "fieldPath": { + "type": "string" + } + }, + "type": "object" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + }, + "resourceFieldRef": { + "properties": { + "containerName": { + "type": "string" + }, + "divisor": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "resource": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "name": { + "type": "string" + }, + "optional": { + "type": "boolean" + } + }, + "type": "object" + }, + "serviceAccountToken": { + "properties": { + "audience": { + "type": "string" + }, + "expirationSeconds": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + } + }, + "type": "object" + }, + "quobyte": { + "properties": { + "group": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "registry": { + "type": "string" + }, + "tenant": { + "type": "string" + }, + "user": { + "type": "string" + }, + "volume": { + "type": "string" + } + }, + "type": "object" + }, + "rbd": { + "properties": { + "fsType": { + "type": "string" + }, + "image": { + "type": "string" + }, + "keyring": { + "type": "string" + }, + "monitors": { + "items": { + "type": "string" + }, + "type": "array" + }, + "pool": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "scaleIO": { + "properties": { + "fsType": { + "type": "string" + }, + "gateway": { + "type": "string" + }, + "protectionDomain": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "sslEnabled": { + "type": "boolean" + }, + "storageMode": { + "type": "string" + }, + "storagePool": { + "type": "string" + }, + "system": { + "type": "string" + }, + "volumeName": { + "type": "string" + } + }, + "type": "object" + }, + "secret": { + "properties": { + "defaultMode": { + "type": "integer" + }, + "items": { + "items": { + "properties": { + "key": { + "type": "string" + }, + "mode": { + "type": "integer" + }, + "path": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + "optional": { + "type": "boolean" + }, + "secretName": { + "type": "string" + } + }, + "type": "object" + }, + "storageos": { + "properties": { + "fsType": { + "type": "string" + }, + "readOnly": { + "type": "boolean" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "volumeName": { + "type": "string" + }, + "volumeNamespace": { + "type": "string" + } + }, + "type": "object" + }, + "vsphereVolume": { + "properties": { + "fsType": { + "type": "string" + }, + "storagePolicyID": { + "type": "string" + }, + "storagePolicyName": { + "type": "string" + }, + "volumePath": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "required": [ + "containers" + ], + "type": "object" + } + }, + "required": [ + "labels", + "annotations", + "spec" + ], + "type": "object" + }, + "priorityClassName": { + "type": "string" + }, + "readinessProbe": { + "properties": { + "failureThreshold": { + "type": "integer" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "successThreshold": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + }, + "required": [ + "initialDelaySeconds", + "failureThreshold", + "periodSeconds" + ], + "type": "object" + }, + "replicas": { + "type": "integer" + }, + "securityContext": { + "deprecated": true, + "properties": { + "allowPriviledgeEscalation": { + "type": "boolean" + }, + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "fsGroup": { + "type": "integer" + }, + "fsGroupChangePolicy": { + "enum": [ + "OnRootMismatch", + "Always" + ], + "type": "string" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + } + }, + "type": "object" + }, + "sideCars": { + "properties": { + "configWatcher": { + "properties": { + "enabled": { + "type": "boolean" + }, + "extraVolumeMounts": { + "type": "string" + }, + "resources": { + "type": "object" + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "drop": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "controllers": { + "properties": { + "createRBAC": { + "type": "boolean" + }, + "enabled": { + "type": "boolean" + }, + "healthProbeAddress": { + "type": "string" + }, + "image": { + "properties": { + "repository": { + "default": "docker.redpanda.com/redpandadata/redpanda-operator", + "type": "string" + }, + "tag": { + "default": "Chart.appVersion", + "pattern": "^v(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$|^$", + "type": "string" + } + }, + "required": [ + "tag", + "repository" + ], + "type": "object" + }, + "metricsAddress": { + "type": "string" + }, + "resources": true, + "run": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "securityContext": { + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + }, + "capabilities": { + "properties": { + "add": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "drop": { + "oneOf": [ + { + "items": { + "type": "string" + }, + "type": "array" + }, + { + "type": "null" + } + ] + } + }, + "type": "object" + }, + "privileged": { + "type": "boolean" + }, + "procMount": { + "type": "string" + }, + "readOnlyRootFilesystem": { + "type": "boolean" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + }, + "seLinuxOptions": { + "properties": { + "level": { + "type": "string" + }, + "role": { + "type": "string" + }, + "type": { + "type": "string" + }, + "user": { + "type": "string" + } + }, + "type": "object" + }, + "seccompProfile": { + "properties": { + "localhostProfile": { + "type": "string" + }, + "type": { + "type": "string" + } + }, + "type": "object" + }, + "windowsOptions": { + "properties": { + "gmsaCredentialSpec": { + "type": "string" + }, + "gmsaCredentialSpecName": { + "type": "string" + }, + "hostProcess": { + "type": "boolean" + }, + "runAsUserName": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "startupProbe": { + "properties": { + "failureThreshold": { + "type": "integer" + }, + "initialDelaySeconds": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + } + }, + "required": [ + "initialDelaySeconds", + "failureThreshold", + "periodSeconds" + ], + "type": "object" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "tolerations": { + "oneOf": [ + { + "items": { + "properties": { + "effect": { + "type": "string" + }, + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "tolerationSeconds": { + "type": "integer" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "topologySpreadConstraints": { + "oneOf": [ + { + "items": { + "properties": { + "maxSkew": { + "type": "integer" + }, + "topologyKey": { + "type": "string" + }, + "whenUnsatisfiable": { + "pattern": "^(ScheduleAnyway|DoNotSchedule)$", + "type": "string" + } + }, + "type": "object" + }, + "minItems": 1, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "updateStrategy": { + "properties": { + "type": { + "pattern": "^(RollingUpdate|OnDelete)$", + "type": "string" + } + }, + "required": [ + "type" + ], + "type": "object" + } + }, + "required": [ + "additionalSelectorLabels", + "replicas", + "updateStrategy", + "podTemplate", + "budget", + "startupProbe", + "livenessProbe", + "readinessProbe", + "podAffinity", + "podAntiAffinity", + "nodeSelector", + "priorityClassName", + "topologySpreadConstraints", + "tolerations", + "securityContext", + "sideCars" + ], + "type": "object" + }, + "storage": { + "properties": { + "hostPath": { + "type": "string" + }, + "persistentVolume": { + "deprecated": true, + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "nameOverwrite": { + "type": "string" + }, + "size": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "storageClass": { + "type": "string" + } + }, + "required": [ + "annotations", + "enabled", + "labels", + "size", + "storageClass" + ], + "type": "object" + }, + "tiered": { + "properties": { + "config": { + "properties": { + "cloud_storage_access_key": { + "type": "string" + }, + "cloud_storage_api_endpoint": { + "type": "string" + }, + "cloud_storage_api_endpoint_port": { + "type": "integer" + }, + "cloud_storage_azure_adls_endpoint": { + "type": "string" + }, + "cloud_storage_azure_adls_port": { + "type": "integer" + }, + "cloud_storage_bucket": { + "type": "string" + }, + "cloud_storage_cache_check_interval": { + "type": "integer" + }, + "cloud_storage_cache_directory": { + "type": "string" + }, + "cloud_storage_cache_size": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "cloud_storage_credentials_source": { + "pattern": "^(config_file|aws_instance_metadata|sts|gcp_instance_metadata)$", + "type": "string" + }, + "cloud_storage_disable_tls": { + "type": "boolean" + }, + "cloud_storage_enable_remote_read": { + "type": "boolean" + }, + "cloud_storage_enable_remote_write": { + "type": "boolean" + }, + "cloud_storage_enabled": { + "type": "boolean" + }, + "cloud_storage_initial_backoff_ms": { + "type": "integer" + }, + "cloud_storage_manifest_upload_timeout_ms": { + "type": "integer" + }, + "cloud_storage_max_connection_idle_time_ms": { + "type": "integer" + }, + "cloud_storage_max_connections": { + "type": "integer" + }, + "cloud_storage_reconciliation_interval_ms": { + "type": "integer" + }, + "cloud_storage_region": { + "type": "string" + }, + "cloud_storage_secret_key": { + "type": "string" + }, + "cloud_storage_segment_max_upload_interval_sec": { + "type": "integer" + }, + "cloud_storage_segment_upload_timeout_ms": { + "type": "integer" + }, + "cloud_storage_trust_file": { + "type": "string" + }, + "cloud_storage_upload_ctrl_d_coeff": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_max_shares": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_min_shares": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_p_coeff": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_update_interval_ms": { + "type": "integer" + } + }, + "required": [ + "cloud_storage_enabled" + ], + "type": "object" + }, + "credentialsSecretRef": { + "properties": { + "accessKey": { + "properties": { + "configurationKey": { + "type": "string" + }, + "key": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "secretKey": { + "properties": { + "configurationKey": { + "type": "string" + }, + "key": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "type": "object" + }, + "hostPath": { + "type": "string" + }, + "mountType": { + "pattern": "^(none|hostPath|emptyDir|persistentVolume)$", + "type": "string" + }, + "persistentVolume": { + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "nameOverwrite": { + "type": "string" + }, + "size": { + "type": "string" + }, + "storageClass": { + "type": "string" + } + }, + "required": [ + "annotations", + "labels", + "storageClass" + ], + "type": "object" + } + }, + "required": [ + "mountType" + ], + "type": "object" + }, + "tieredConfig": { + "deprecated": true, + "properties": { + "cloud_storage_access_key": { + "type": "string" + }, + "cloud_storage_api_endpoint": { + "type": "string" + }, + "cloud_storage_api_endpoint_port": { + "type": "integer" + }, + "cloud_storage_azure_adls_endpoint": { + "type": "string" + }, + "cloud_storage_azure_adls_port": { + "type": "integer" + }, + "cloud_storage_bucket": { + "type": "string" + }, + "cloud_storage_cache_check_interval": { + "type": "integer" + }, + "cloud_storage_cache_directory": { + "type": "string" + }, + "cloud_storage_cache_size": { + "oneOf": [ + { + "type": "integer" + }, + { + "pattern": "^[0-9]+(\\.[0-9]){0,1}(m|k|M|G|T|P|Ki|Mi|Gi|Ti|Pi)?$", + "type": "string" + } + ] + }, + "cloud_storage_credentials_source": { + "pattern": "^(config_file|aws_instance_metadata|sts|gcp_instance_metadata)$", + "type": "string" + }, + "cloud_storage_disable_tls": { + "type": "boolean" + }, + "cloud_storage_enable_remote_read": { + "type": "boolean" + }, + "cloud_storage_enable_remote_write": { + "type": "boolean" + }, + "cloud_storage_enabled": { + "type": "boolean" + }, + "cloud_storage_initial_backoff_ms": { + "type": "integer" + }, + "cloud_storage_manifest_upload_timeout_ms": { + "type": "integer" + }, + "cloud_storage_max_connection_idle_time_ms": { + "type": "integer" + }, + "cloud_storage_max_connections": { + "type": "integer" + }, + "cloud_storage_reconciliation_interval_ms": { + "type": "integer" + }, + "cloud_storage_region": { + "type": "string" + }, + "cloud_storage_secret_key": { + "type": "string" + }, + "cloud_storage_segment_max_upload_interval_sec": { + "type": "integer" + }, + "cloud_storage_segment_upload_timeout_ms": { + "type": "integer" + }, + "cloud_storage_trust_file": { + "type": "string" + }, + "cloud_storage_upload_ctrl_d_coeff": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_max_shares": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_min_shares": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_p_coeff": { + "type": "integer" + }, + "cloud_storage_upload_ctrl_update_interval_ms": { + "type": "integer" + } + }, + "type": "object" + }, + "tieredStorageHostPath": { + "deprecated": true, + "type": "string" + }, + "tieredStoragePersistentVolume": { + "deprecated": true, + "properties": { + "annotations": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "labels": { + "additionalProperties": { + "type": "string" + }, + "type": "object" + }, + "storageClass": { + "type": "string" + } + }, + "required": [ + "annotations", + "enabled", + "labels", + "storageClass" + ], + "type": "object" + } + }, + "required": [ + "hostPath", + "tiered", + "persistentVolume" + ], + "type": "object" + }, + "tests": { + "properties": { + "enabled": { + "type": "boolean" + } + }, + "type": "object" + }, + "tls": { + "properties": { + "certs": { + "minProperties": 1, + "patternProperties": { + "^[A-Za-z_][A-Za-z0-9_]*$": { + "properties": { + "applyInternalDNSNames": { + "type": "boolean" + }, + "caEnabled": { + "type": "boolean" + }, + "clientSecretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + }, + "duration": { + "pattern": ".*[smh]$", + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "issuerRef": { + "properties": { + "group": { + "type": "string" + }, + "kind": { + "type": "string" + }, + "name": { + "type": "string" + } + }, + "type": "object" + }, + "secretRef": { + "properties": { + "name": { + "type": "string" + } + }, + "type": "object" + } + }, + "required": [ + "caEnabled" + ], + "type": "object" + } + }, + "type": "object" + }, + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled", + "certs" + ], + "type": "object" + }, + "tolerations": { + "oneOf": [ + { + "items": { + "properties": { + "effect": { + "type": "string" + }, + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "tolerationSeconds": { + "type": "integer" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + "type": "array" + }, + { + "type": "null" + } + ] + }, + "tuning": { + "properties": { + "ballast_file_path": { + "type": "string" + }, + "ballast_file_size": { + "type": "string" + }, + "tune_aio_events": { + "type": "boolean" + }, + "tune_ballast_file": { + "type": "boolean" + }, + "tune_clocksource": { + "type": "boolean" + }, + "well_known_io": { + "type": "string" + } + }, + "type": "object" + } + }, + "required": [ + "affinity", + "image" + ], + "type": "object" +} diff --git a/charts/redpanda/redpanda/5.9.10/values.yaml b/charts/redpanda/redpanda/5.9.10/values.yaml new file mode 100644 index 0000000000..2d23d19469 --- /dev/null +++ b/charts/redpanda/redpanda/5.9.10/values.yaml @@ -0,0 +1,1133 @@ +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# This file contains values for variables referenced from yaml files in the templates directory. +# +# For further information on Helm templating see the documentation at: +# https://helm.sh/docs/chart_template_guide/values_files/ + +# +# >>> This chart requires Helm version 3.6.0 or greater <<< +# + +# Common settings +# +# -- Override `redpanda.name` template. +nameOverride: "" +# -- Override `redpanda.fullname` template. +fullnameOverride: "" +# -- Default Kubernetes cluster domain. +clusterDomain: cluster.local +# -- Additional labels to add to all Kubernetes objects. +# For example, `my.k8s.service: redpanda`. +commonLabels: {} +# -- Node selection constraints for scheduling Pods, can override this for StatefulSets. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). +nodeSelector: {} +# -- Affinity constraints for scheduling Pods, can override this for StatefulSets and Jobs. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#affinity-and-anti-affinity). +affinity: {} +# -- Taints to be tolerated by Pods, can override this for StatefulSets. +# For details, +# see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). +tolerations: [] + +# -- Redpanda Docker image settings. +image: + # -- Docker repository from which to pull the Redpanda Docker image. + repository: docker.redpanda.com/redpandadata/redpanda + # -- The Redpanda version. + # See DockerHub for: + # [All stable versions](https://hub.docker.com/r/redpandadata/redpanda/tags) + # and [all unstable versions](https://hub.docker.com/r/redpandadata/redpanda-unstable/tags). + # @default -- `Chart.appVersion`. + tag: "" + # -- The imagePullPolicy. + # If `image.tag` is 'latest', the default is `Always`. + pullPolicy: IfNotPresent + +# -- Redpanda Service settings. +# service: +# -- set service.name to override the default service name +# name: redpanda +# -- internal Service +# internal: +# -- add annotations to the internal Service +# annotations: {} +# +# -- eg. for a bare metal install using external-dns +# annotations: +# "external-dns.alpha.kubernetes.io/hostname": redpanda.domain.dom +# "external-dns.alpha.kubernetes.io/endpoints-type": HostIP + +# -- Pull secrets may be used to provide credentials to image repositories +# See the [Kubernetes documentation](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). +imagePullSecrets: [] + +# -- DEPRECATED Enterprise license key (optional). +# For details, +# see the [License documentation](https://docs.redpanda.com/docs/get-started/licenses/?platform=kubernetes#redpanda-enterprise-edition). +license_key: "" +# -- DEPRECATED Secret name and secret key where the license key is stored. +license_secret_ref: {} + # secret_name: my-secret + # secret_key: key-where-license-is-stored + +# -- Audit logging for a redpanda cluster, must have enabled sasl and have one kafka listener supporting sasl authentication +# for audit logging to work. Note this feature is only available for redpanda versions >= v23.3.0. +auditLogging: + # -- Enable or disable audit logging, for production clusters we suggest you enable, + # however, this will only work if you also enable sasl and a listener with sasl enabled. + enabled: false + # -- Kafka listener name, note that it must have `authenticationMethod` set to `sasl`. + # For external listeners, use the external listener name, such as `default`. + listener: internal + # -- Integer value defining the number of partitions used by a newly created audit topic. + partitions: 12 + # -- Event types that should be captured by audit logs, default is [`admin`, `authenticate`, `management`]. + enabledEventTypes: + # -- List of topics to exclude from auditing, default is null. + excludedTopics: + # -- List of principals to exclude from auditing, default is null. + excludedPrincipals: + # -- Defines the number of bytes (in bytes) allocated by the internal audit client for audit messages. + clientMaxBufferSize: 16777216 + # -- In ms, frequency in which per shard audit logs are batched to client for write to audit log. + queueDrainIntervalMs: 500 + # -- Defines the maximum amount of memory used (in bytes) by the audit buffer in each shard. + queueMaxBufferSizePerShard: 1048576 + # -- Defines the replication factor for a newly created audit log topic. This configuration applies + # only to the audit log topic and may be different from the cluster or other topic configurations. + # This cannot be altered for existing audit log topics. Setting this value is optional. If a value is not provided, + # Redpanda will use the `internal_topic_replication_factor cluster` config value. Default is `null` + replicationFactor: + +# -- Enterprise (optional) +# For details, +# see the [License documentation](https://docs.redpanda.com/docs/get-started/licenses/?platform=kubernetes#redpanda-enterprise-edition). +enterprise: + # -- license (optional). + license: "" + # -- Secret name and key where the license key is stored. + licenseSecretRef: {} + # name: my-secret + # key: key-where-license-is-stored + +# -- Rack Awareness settings. +# For details, +# see the [Rack Awareness documentation](https://docs.redpanda.com/docs/manage/kubernetes/kubernetes-rack-awareness/). +rackAwareness: + # -- When running in multiple racks or availability zones, use a Kubernetes Node + # annotation value as the Redpanda rack value. + # Enabling this requires running with a service account with "get" Node permissions. + # To have the Helm chart configure these permissions, + # set `serviceAccount.create=true` and `rbac.enabled=true`. + enabled: false + # -- The common well-known annotation to use as the rack ID. + # Override this only if you use a custom Node annotation. + nodeAnnotation: topology.kubernetes.io/zone + +# +# -- Redpanda Console settings. +# For a reference of configuration settings, +# see the [Redpanda Console documentation](https://docs.redpanda.com/docs/reference/console/config/). +console: + enabled: true + configmap: + create: false + secret: + create: false + deployment: + create: false + config: {} + +# +# -- Redpanda Managed Connectors settings +# For a reference of configuration settings, +# see the [Redpanda Connectors documentation](https://docs.redpanda.com/docs/deploy/deployment-option/cloud/managed-connectors/). +connectors: + enabled: false + deployment: + create: false + test: + create: false + +# -- Authentication settings. +# For details, +# see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/). +auth: + sasl: + # -- Enable SASL authentication. + # If you enable SASL authentication, you must provide a Secret in `auth.sasl.secretRef`. + enabled: false + # -- The authentication mechanism to use for the superuser. Options are `SCRAM-SHA-256` and `SCRAM-SHA-512`. + mechanism: SCRAM-SHA-512 + # -- A Secret that contains your superuser credentials. + # For details, + # see the [SASL documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/sasl-kubernetes/#use-secrets). + secretRef: "redpanda-users" + # -- Optional list of superusers. + # These superusers will be created in the Secret whose name is defined in `auth.sasl.secretRef`. + # If this list is empty, + # the Secret in `auth.sasl.secretRef` must already exist in the cluster before you deploy the chart. + # Uncomment the sample list if you wish to try adding sample sasl users or override to use your own. + users: [] + # - name: admin + # password: change-me + # mechanism: SCRAM-SHA-512 + # -- Details about how to create the bootstrap user for the cluster. + # The secretKeyRef is optionally specified. If it is specified, the + # chart will use a password written to that secret when creating the + # "kubernetes-controller" bootstrap user. If it is unspecified, then + # the secret will be generated and stored in the secret + # "releasename"-bootstrap-user, with the key "password". + bootstrapUser: + # -- The name used to override the name of the bootstrap user. If unspecified the bootstrap user is named + # "kubernetes-controller". This should only be specified when SASL authentication is enabled (usually installation) + # and should not be changed afterward. + # name: my-user + # -- The authentication mechanism to use for the bootstrap user. Options are `SCRAM-SHA-256` and `SCRAM-SHA-512`. + mechanism: SCRAM-SHA-256 + # secretKeyRef: + # name: my-password + # key: my-key + +# -- TLS settings. +# For details, see the [TLS documentation](https://docs.redpanda.com/docs/manage/kubernetes/security/kubernetes-tls/). +tls: + # -- Enable TLS globally for all listeners. + # Each listener must include a Certificate name in its `.tls` object. + # To allow you to enable TLS for individual listeners, + # Certificates in `auth.tls.certs` are always loaded, even if `tls.enabled` is `false`. + # See `listeners..tls.enabled`. + enabled: true + # -- List all Certificates here, + # then you can reference a specific Certificate's name + # in each listener's `listeners..tls.cert` setting. + certs: + # -- This key is the Certificate name. + # To apply the Certificate to a specific listener, + # reference the Certificate's name in `listeners..tls.cert`. + default: + # -- To use a custom pre-installed Issuer, + # add its name and kind to the `issuerRef` object. + # issuerRef: + # name: redpanda-default-root-issuer + # kind: Issuer # Can be Issuer or ClusterIssuer + # -- To use a secret with custom tls files, + # secretRef: + # name: my-tls-secret + # -- Indicates whether or not the Secret holding this certificate + # includes a `ca.crt` key. When `true`, chart managed clients, such as + # rpk, will use `ca.crt` for certificate verification and listeners with + # `require_client_auth` and no explicit `truststore` will use `ca.crt` as + # their `truststore_file` for verification of client certificates. When + # `false`, chart managed clients will use `tls.crt` for certificate + # verification and listeners with `require_client_auth` and no explicit + # `truststore` will use the container's CA certificates. + caEnabled: true + # duration: 43800h + # if you wish to have Kubernetes internal dns names (IE the headless service of the redpanda StatefulSet) included in `dnsNames` of the certificate even, when supplying an issuer. + # applyInternalDNSNames: false + # -- Example external tls configuration + # uncomment and set the right key to the listeners that require them + # also enable the tls setting for those listeners. + external: + # -- To use a custom pre-installed Issuer, + # add its name and kind to the `issuerRef` object. + # issuerRef: + # name: redpanda-default-root-issuer + # kind: Issuer # Can be Issuer or ClusterIssuer + # -- To use a secret with custom tls files, + # secretRef: + # name: my-tls-secret + # -- Indicates whether or not the Secret holding this certificate + # includes a `ca.crt` key. When `true`, chart managed clients, such as + # rpk, will use `ca.crt` for certificate verification and listeners with + # `require_client_auth` and no explicit `truststore` will use `ca.crt` as + # their `truststore_file` for verification of client certificates. When + # `false`, chart managed clients will use `tls.crt` for certificate + # verification and listeners with `require_client_auth` and no explicit + # `truststore` will use the container's CA certificates. + caEnabled: true + # duration: 43800h + # if you wish to for apply internal dns names to the certificate even when supplying an issuer + # applyInternalDNSNames: false + +# -- External access settings. +# For details, +# see the [Networking and Connectivity documentation](https://docs.redpanda.com/docs/manage/kubernetes/networking/networking-and-connectivity/). +external: + # -- Service allows you to manage the creation of an external kubernetes service object + service: + # -- Enabled if set to false will not create the external service type + # You can still set your cluster with external access but not create the supporting service (NodePort/LoadBalander). + # Set this to false if you rather manage your own service. + enabled: true + # -- Enable external access for each Service. + # You can toggle external access for each listener in + # `listeners..external..enabled`. + enabled: true + # -- External access type. Only `NodePort` and `LoadBalancer` are supported. + # If undefined, then advertised listeners will be configured in Redpanda, + # but the helm chart will not create a Service. + # You must create a Service manually. + # Warning: If you use LoadBalancers, you will likely experience higher latency and increased packet loss. + # NodePort is recommended in cases where latency is a priority. + type: NodePort + # Optional source range for external access. Only applicable when external.type is LoadBalancer + # sourceRanges: [] + # -- Optional domain advertised to external clients + # If specified, then it will be appended to the `external.addresses` values as each broker's advertised address + # domain: local + # Optional list of addresses that the Redpanda brokers advertise. + # Provide one entry for each broker in order of StatefulSet replicas. + # The number of brokers is defined in statefulset.replicas. + # The values can be IP addresses or DNS names. + # If external.domain is set, the domain is appended to these values. + # There is an option to define a single external address for all brokers and leverage + # prefixTemplate as it will be calculated during initContainer execution. + # addresses: + # - redpanda-0 + # - redpanda-1 + # - redpanda-2 + # + # annotations: + # For example: + # cloud.google.com/load-balancer-type: "Internal" + # service.beta.kubernetes.io/aws-load-balancer-type: nlb + # If you enable externalDns, each LoadBalancer service instance + # will be annotated with external-dns hostname + # matching external.addresses + external.domain + # externalDns: + # enabled: true + # prefixTemplate: "" + +# -- Log-level settings. +logging: + # -- Log level + # Valid values (from least to most verbose) are: `warn`, `info`, `debug`, and `trace`. + logLevel: info + # -- Send usage statistics back to Redpanda Data. + # For details, + # see the [stats reporting documentation](https://docs.redpanda.com/docs/cluster-administration/monitoring/#stats-reporting). + usageStats: + # Enable the `rpk.enable_usage_stats` property. + enabled: true + # Your cluster ID (optional) + # clusterId: your-helm-cluster + +# -- Monitoring. +# This will create a ServiceMonitor that can be used by Prometheus-Operator or VictoriaMetrics-Operator to scrape the metrics. +monitoring: + enabled: false + scrapeInterval: 30s + labels: {} + # Enables http2 for scraping metrics for prometheus. Used when Istio's mTLS is enabled and using tlsConfig. + # enableHttp2: true + # tlsConfig: + # caFile: /etc/prom-certs/root-cert.pem + # certFile: /etc/prom-certs/cert-chain.pem + # insecureSkipVerify: true + # keyFile: /etc/prom-certs/key.pem + +# -- Pod resource management. +# This section simplifies resource allocation +# by providing a single location where resources are defined. +# Helm sets these resource values within the `statefulset.yaml` and `configmap.yaml` templates. +# +# The default values are for a development environment. +# Production-level values and other considerations are documented, +# where those values are different from the default. +# For details, +# see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/). +resources: + # + # -- CPU resources. + # For details, + # see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/#configure-cpu-resources). + cpu: + # -- Redpanda makes use of a thread per core model. + # For details, see this [blog](https://redpanda.com/blog/tpc-buffers). + # For this reason, Redpanda should only be given full cores. + # + # Note: You can increase cores, but decreasing cores is not currently supported. + # See the [GitHub issue](https://github.com/redpanda-data/redpanda/issues/350). + # + # This setting is equivalent to `--smp`, `resources.requests.cpu`, and `resources.limits.cpu`. + # For production, use `4` or greater. + # + # To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for + # CPU resource requests and limits. This policy gives the Pods running Redpanda brokers + # access to exclusive CPUs on the node. See + # https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy. + cores: 1 + # + # -- Overprovisioned means Redpanda won't assume it has all of the provisioned CPU. + # This should be true unless the container has CPU affinity. + # Equivalent to: `--idle-poll-time-us 0 --thread-affinity 0 --poll-aio 0` + # + # If the value of full cores in `resources.cpu.cores` is less than `1`, this + # setting is set to `true`. + # overprovisioned: false + # + # -- Memory resources + # For details, + # see the [Pod resources documentation](https://docs.redpanda.com/docs/manage/kubernetes/manage-resources/#configure-memory-resources). + memory: + # -- Enables memory locking. + # For production, set to `true`. + # enable_memory_locking: false + # + # It is recommended to have at least 2Gi of memory per core for the Redpanda binary. + # This memory is taken from the total memory given to each container. + # The Helm chart allocates 80% of the container's memory to Redpanda, leaving the rest for + # the Seastar subsystem (reserveMemory) and other container processes. + # So at least 2.5Gi per core is recommended in order to ensure Redpanda has a full 2Gi. + # + # These values affect `--memory` and `--reserve-memory` flags passed to Redpanda and the memory + # requests/limits in the StatefulSet. + # Valid suffixes: k, M, G, T, P, Ki, Mi, Gi, Ti, Pi + # To create `Guaranteed` Pod QoS for Redpanda brokers, provide both container max and min values for the container. + # For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a memory limit and a memory request. + # * For every container in the Pod, the memory limit must equal the memory request. + # + container: + # Minimum memory count for each Redpanda broker. + # If omitted, the `min` value is equal to the `max` value (requested resources defaults to limits). + # This setting is equivalent to `resources.requests.memory`. + # For production, use 10Gi or greater. + # min: 2.5Gi + # + # -- Maximum memory count for each Redpanda broker. + # Equivalent to `resources.limits.memory`. + # For production, use `10Gi` or greater. + max: 2.5Gi + # + # This optional `redpanda` object allows you to specify the memory size for both the Redpanda + # process and the underlying reserved memory used by Seastar. + # This section is omitted by default, and memory sizes are calculated automatically + # based on container memory. + # Uncommenting this section and setting memory and reserveMemory values will disable + # automatic calculation. + # + # If you are setting the following values manually, keep in mind the following guidelines. + # Getting this wrong may lead to performance issues, instability, and loss of data: + # The amount of memory to allocate to a container is determined by the sum of three values: + # 1. Redpanda (at least 2Gi per core, ~80% of the container's total memory) + # 2. Seastar subsystem (200Mi * 0.2% of the container's total memory, 200Mi < x < 1Gi) + # 3. Other container processes (whatever small amount remains) + # redpanda: + # Memory for the Redpanda process. + # This must be lower than the container's memory (resources.memory.container.min if provided, otherwise + # resources.memory.container.max). + # Equivalent to --memory. + # For production, use 8Gi or greater. + # memory: 2Gi + # + # Memory reserved for the Seastar subsystem. + # Any value above 1Gi will provide diminishing performance benefits. + # Equivalent to --reserve-memory. + # For production, use 1Gi. + # reserveMemory: 200Mi + +# -- Persistence settings. +# For details, see the [storage documentation](https://docs.redpanda.com/docs/manage/kubernetes/configure-storage/). +storage: + # -- Absolute path on the host to store Redpanda's data. + # If unspecified, then an `emptyDir` volume is used. + # If specified but `persistentVolume.enabled` is true, `storage.hostPath` has no effect. + hostPath: "" + # -- If `persistentVolume.enabled` is true, a PersistentVolumeClaim is created and + # used to store Redpanda's data. Otherwise, `storage.hostPath` is used. + persistentVolume: + enabled: true + size: 20Gi + # -- To disable dynamic provisioning, set to `-`. + # If undefined or empty (default), then no storageClassName spec is set, + # and the default dynamic provisioner is chosen (gp2 on AWS, standard on + # GKE, AWS & OpenStack). + storageClass: "" + # -- Additional labels to apply to the created PersistentVolumeClaims. + labels: {} + # -- Additional annotations to apply to the created PersistentVolumeClaims. + annotations: {} + # -- Option to change volume claim template name for tiered storage persistent volume + # if tiered.mountType is set to `persistentVolume` + nameOverwrite: "" + # + # Settings for the Tiered Storage cache. + # For details, + # see the [Tiered Storage documentation](https://docs.redpanda.com/docs/manage/kubernetes/tiered-storage/#caching). + + tiered: + # mountType can be one of: + # - none: does not mount a volume. Tiered storage will use the data directory. + # - hostPath: will allow you to chose a path on the Node the pod is running on + # - emptyDir: will mount a fresh empty directory every time the pod starts + # - persistentVolume: creates and mounts a PersistentVolumeClaim + mountType: emptyDir + + # For the maximum size of the disk cache, see `tieredConfig.cloud_storage_cache_size`. + # + # -- Absolute path on the host to store Redpanda's Tiered Storage cache. + hostPath: "" + # PersistentVolumeClaim to be created for the Tiered Storage cache and + # used to store data retrieved from cloud storage, such as S3). + persistentVolume: + # -- To disable dynamic provisioning, set to "-". + # If undefined or empty (default), then no storageClassName spec is set, + # and the default dynamic provisioner is chosen (gp2 on AWS, standard on + # GKE, AWS & OpenStack). + storageClass: "" + # -- Additional labels to apply to the created PersistentVolumeClaims. + labels: {} + # -- Additional annotations to apply to the created PersistentVolumeClaims. + annotations: {} + + # credentialsSecretRef can be used to set `cloud_storage_secret_key` and/or `cloud_storage_access_key` from + # referenced Kubernetes Secret + credentialsSecretRef: + accessKey: + # https://docs.redpanda.com/current/reference/object-storage-properties/#cloud_storage_access_key + configurationKey: cloud_storage_access_key + # name: + # key: + secretKey: + # https://docs.redpanda.com/current/reference/object-storage-properties/#cloud_storage_secret_key + # or + # https://docs.redpanda.com/current/reference/object-storage-properties/#cloud_storage_azure_shared_key + configurationKey: cloud_storage_secret_key + # name: + # key + # -- DEPRECATED `configurationKey`, `name` and `key`. Please use `accessKey` and `secretKey` + # configurationKey: cloud_storage_secret_key + # name: + # key: + # + # -- Tiered Storage settings + # Requires `enterprise.licenseKey` or `enterprised.licenseSecretRef` + # For details, see the [Tiered Storage documentation](https://docs.redpanda.com/docs/manage/kubernetes/tiered-storage/). + # For a list of properties, see [Object Storage Properties](https://docs.redpanda.com/current/reference/properties/object-storage-properties/). + config: + # -- Global flag that enables Tiered Storage if a license key is provided. + # See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enabled). + cloud_storage_enabled: false + # -- Cluster level default remote write configuration for new topics. + # See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enable_remote_write). + cloud_storage_enable_remote_write: true + # -- Cluster level default remote read configuration for new topics. + # See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_enable_remote_read). + cloud_storage_enable_remote_read: true + # -- Maximum size of the disk cache used by Tiered Storage. + # Default is 20 GiB. + # See the [property reference documentation](https://docs.redpanda.com/docs/reference/object-storage-properties/#cloud_storage_cache_size). + cloud_storage_cache_size: 5368709120 + +post_install_job: + enabled: true + # Resource requests and limits for the post-install batch job + # resources: + # requests: + # cpu: 1 + # memory: 512Mi + # limits: + # cpu: 2 + # memory: 1024Mi + # labels: {} + # annotations: {} + affinity: {} + + podTemplate: + # -- Additional labels to apply to the Pods of this Job. + labels: {} + # -- Additional annotations to apply to the Pods of this Job. + annotations: {} + # -- A subset of Kubernetes' PodSpec type that will be merged into the + # final PodSpec. See [Merge Semantics](#merging-semantics) for details. + spec: + securityContext: {} + containers: + - name: post-install + securityContext: {} + env: [] + +statefulset: + # -- Number of Redpanda brokers (Redpanda Data recommends setting this to the number of worker nodes in the cluster) + replicas: 3 + updateStrategy: + type: RollingUpdate + budget: + maxUnavailable: 1 + # -- DEPRECATED Please use statefulset.podTemplate.annotations. + # Annotations are used only for `Statefulset.spec.template.metadata.annotations`. The StatefulSet does not have + # any dedicated annotation. + annotations: {} + # -- Additional labels to be added to statefulset label selector. + # For example, `my.k8s.service: redpanda`. + additionalSelectorLabels: {} + podTemplate: + # -- Additional labels to apply to the Pods of the StatefulSet. + labels: {} + # -- Additional annotations to apply to the Pods of the StatefulSet. + annotations: {} + # -- A subset of Kubernetes' PodSpec type that will be merged into the + # final PodSpec. See [Merge Semantics](#merging-semantics) for details. + spec: + securityContext: {} + containers: + - name: redpanda + securityContext: {} + env: [] + # -- Adjust the period for your probes to meet your needs. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#container-probes). + startupProbe: + initialDelaySeconds: 1 + failureThreshold: 120 + periodSeconds: 10 + livenessProbe: + initialDelaySeconds: 10 + failureThreshold: 3 + periodSeconds: 10 + readinessProbe: + initialDelaySeconds: 1 + failureThreshold: 3 + periodSeconds: 10 + successThreshold: 1 + # + # StatefulSet resources: + # Resources are set through the top-level resources section above. + # It is recommended to set resource values in that section rather than here, as this will guarantee + # memory is allocated across containers, Redpanda, and the Seastar subsystem correctly. + # This automatic memory allocation is in place because Repanda and the Seastar subsystem require flags + # at startup that set the amount of memory available to each process. + # Kubernetes (mainly statefulset), Redpanda, and Seastar memory values are tightly coupled. + # Adding a resource section here will be ignored. + # + # -- Inter-Pod Affinity rules for scheduling Pods of this StatefulSet. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + podAffinity: {} + # -- Anti-affinity rules for scheduling Pods of this StatefulSet. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#inter-pod-affinity-and-anti-affinity). + # You may either edit the default settings for anti-affinity rules, + # or specify new anti-affinity rules to use instead of the defaults. + podAntiAffinity: + # -- The topologyKey to be used. + # Can be used to spread across different nodes, AZs, regions etc. + topologyKey: kubernetes.io/hostname + # -- Valid anti-affinity types are `soft`, `hard`, or `custom`. + # Use `custom` if you want to supply your own anti-affinity rules in the `podAntiAffinity.custom` object. + type: hard + # -- Weight for `soft` anti-affinity rules. + # Does not apply to other anti-affinity types. + weight: 100 + # -- Change `podAntiAffinity.type` to `custom` and provide your own podAntiAffinity rules here. + custom: {} + # -- Node selection constraints for scheduling Pods of this StatefulSet. + # These constraints override the global `nodeSelector` value. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector). + nodeSelector: {} + # -- PriorityClassName given to Pods of this StatefulSet. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/#priorityclass). + priorityClassName: "" + # -- Taints to be tolerated by Pods of this StatefulSet. + # These tolerations override the global tolerations value. + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/). + tolerations: [] + # For details, + # see the [Kubernetes documentation](https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/). + topologySpreadConstraints: + - maxSkew: 1 + topologyKey: topology.kubernetes.io/zone + whenUnsatisfiable: ScheduleAnyway + # -- DEPRECATED: Prefer to use podTemplate.spec.securityContext or podTemplate.spec.containers[0].securityContext. + securityContext: + fsGroup: 101 + runAsUser: 101 + fsGroupChangePolicy: OnRootMismatch + sideCars: + configWatcher: + enabled: true + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a memory limit and a memory request. + # * For every container in the Pod, the memory limit must equal the memory request. + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + # + # To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for + # CPU resource requests and limits. This policy gives the Pods running Redpanda brokers + # access to exclusive CPUs on the node. For details, see + # https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy + resources: {} + securityContext: {} + extraVolumeMounts: |- + # Configure extra controllers to run as sidecars inside the Pods running Redpanda brokers. + # Available controllers: + # - Decommission Controller: The Decommission Controller ensures smooth scaling down operations. + # This controller is responsible for monitoring changes in the number of StatefulSet replicas and orchestrating + # the decommissioning of brokers when necessary. It also sets the reclaim policy for the decommissioned + # broker's PersistentVolume to `Retain` and deletes the corresponding PersistentVolumeClaim. + # - Node-PVC Controller: The Node-PVC Controller handles the PVCs of deleted brokers. + # By setting the PV Retain policy to retain, it facilitates the rescheduling of brokers to new, healthy nodes when + # an existing node is removed. + controllers: + image: + tag: v2.2.5-24.2.7 + repository: docker.redpanda.com/redpandadata/redpanda-operator + # You must also enable RBAC, `rbac.enabled=true`, to deploy this sidecar + enabled: false + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + # + # To maximize efficiency, use the `static` CPU manager policy by specifying an even integer for + # CPU resource requests and limits. This policy gives the Pods running Redpanda brokers + # access to exclusive CPUs on the node. For details, see + # https://kubernetes.io/docs/tasks/administer-cluster/cpu-management-policies/#static-policy + resources: {} + securityContext: {} + healthProbeAddress: ":8085" + metricsAddress: ":9082" + run: + - all + createRBAC: true + initContainers: + fsValidator: + enabled: false + expectedFS: xfs + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + resources: {} + extraVolumeMounts: |- + tuning: + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + resources: {} + extraVolumeMounts: |- + setDataDirOwnership: + # -- In environments where root is not allowed, you cannot change the ownership of files and directories. + # Enable `setDataDirOwnership` when using default minikube cluster configuration. + enabled: false + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + resources: {} + extraVolumeMounts: |- + setTieredStorageCacheDirOwnership: + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + resources: {} + extraVolumeMounts: |- + configurator: + # -- To create `Guaranteed` Pods for Redpanda brokers, provide both requests and limits for CPU and memory. For details, see + # https://kubernetes.io/docs/tasks/configure-pod-container/quality-service-pod/#create-a-pod-that-gets-assigned-a-qos-class-of-guaranteed + # * Every container in the Pod must have a CPU limit and a CPU request. + # * For every container in the Pod, the CPU limit must equal the CPU request. + resources: {} + extraVolumeMounts: |- + ## Additional init containers + extraInitContainers: |- +# - name: "test-init-container" +# image: "mintel/docker-alpine-bash-curl-jq:latest" +# command: [ "/bin/bash", "-c" ] +# args: +# - | +# set -xe +# echo "Hello World!" + initContainerImage: + repository: busybox + tag: latest + # -- Additional flags to pass to redpanda, + additionalRedpandaCmdFlags: [] +# - --unsafe-bypass-fsync + # -- Termination grace period in seconds is time required to execute preStop hook + # which puts particular Redpanda Pod (process/container) into maintenance mode. + # Before settle down on particular value please put Redpanda under load and perform + # rolling upgrade or rolling restart. That value needs to accommodate two processes: + # * preStop hook needs to put Redpanda into maintenance mode + # * after preStop hook Redpanda needs to handle gracefully SIGTERM signal + # + # Both processes are executed sequentially where preStop hook has hard deadline in the + # middle of terminationGracePeriodSeconds. + # + # REF: + # https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#hook-handler-execution + # https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#pod-termination + terminationGracePeriodSeconds: 90 + ## Additional Volumes that you mount + extraVolumes: |- + ## Additional Volume mounts for redpanda container + extraVolumeMounts: |- + +# -- Service account management. +serviceAccount: + # -- Specifies whether a service account should be created. + create: false + # -- Specifies whether a service account should automount API-Credentials. The token is used in sidecars.controllers + automountServiceAccountToken: false + # -- Annotations to add to the service account. + annotations: {} + # -- The name of the service account to use. + # If not set and `serviceAccount.create` is `true`, + # a name is generated using the `redpanda.fullname` template. + name: "" + +# -- Role Based Access Control. +rbac: + # -- Enable for features that need extra privileges. + # If you use the Redpanda Operator, + # you must deploy it with the `--set rbac.createRPKBundleCRs=true` flag + # to give it the required ClusterRoles. + enabled: false + # -- Annotations to add to the `rbac` resources. + annotations: {} + +# -- Redpanda tuning settings. +# Each is set to their default values in Redpanda. +tuning: + # -- Increase the maximum number of outstanding asynchronous IO operations if the + # current value is below a certain threshold. This allows Redpanda to make as many + # simultaneous IO requests as possible, increasing throughput. + # + # When this option is enabled, Helm creates a privileged container. If your security profile does not allow this, you can disable this container by setting `tune_aio_events` to `false`. + # For more details, see the [tuning documentation](https://docs.redpanda.com/docs/deploy/deployment-option/self-hosted/kubernetes/kubernetes-tune-workers/). + tune_aio_events: true + # + # Syncs NTP + # tune_clocksource: false + # + # Creates a "ballast" file so that, if a Redpanda node runs out of space, + # you can delete the ballast file to allow the node to resume operations and then + # delete a topic or records to reduce the space used by Redpanda. + # tune_ballast_file: false + # + # The path where the ballast file will be created. + # ballast_file_path: "/var/lib/redpanda/data/ballast" + # + # The ballast file size. + # ballast_file_size: "1GiB" + # + # (Optional) The vendor, VM type and storage device type that redpanda will run on, in + # the format ::. This hints to rpk which configuration values it + # should use for the redpanda IO scheduler. + # Some valid values are "gcp:c2-standard-16:nvme", "aws:i3.xlarge:default" + # well_known_io: "" + # + # The following tuning parameters must be false in container environments and will be ignored: + # tune_network + # tune_disk_scheduler + # tune_disk_nomerges + # tune_disk_irq + # tune_fstrim + # tune_cpu + # tune_swappiness + # tune_transparent_hugepages + # tune_coredump + + +# -- Listener settings. +# +# Override global settings configured above for individual +# listeners. +# For details, +# see the [listeners documentation](https://docs.redpanda.com/docs/manage/kubernetes/networking/configure-listeners/). +listeners: + # -- Admin API listener (only one). + admin: + # -- The port for both internal and external connections to the Admin API. + port: 9644 + # -- Optional instrumentation hint - https://kubernetes.io/docs/concepts/services-networking/service/#application-protocol + # appProtocol: + # -- Optional external access settings. + external: + # -- Name of the external listener. + default: + port: 9645 + # Override the global `external.enabled` for only this listener. + # enabled: true + # -- The port advertised to this listener's external clients. + # List one port if you want to use the same port for each broker (would be the case when using NodePort service). + # Otherwise, list the port you want to use for each broker in order of StatefulSet replicas. + # If undefined, `listeners.admin.port` is used. + tls: + # enabled: true + cert: external + advertisedPorts: + - 31644 + # -- Optional TLS section (required if global TLS is enabled) + tls: + # Optional flag to override the global TLS enabled flag. + # enabled: true + # -- Name of the Certificate used for TLS (must match a Certificate name that is registered in tls.certs). + cert: default + # -- If true, the truststore file for this listener is included in the ConfigMap. + requireClientAuth: false + # -- Kafka API listeners. + kafka: + # -- The port for internal client connections. + port: 9093 + # default is "sasl" + authenticationMethod: + tls: + # Optional flag to override the global TLS enabled flag. + # enabled: true + cert: default + requireClientAuth: false + external: + default: + # enabled: true + # -- The port used for external client connections. + port: 9094 + # prefixTemplate: "" + # -- If undefined, `listeners.kafka.external.default.port` is used. + advertisedPorts: + - 31092 + tls: + # enabled: true + cert: external + # default is "sasl" + authenticationMethod: + # -- RPC listener (this is never externally accessible). + rpc: + port: 33145 + tls: + # Optional flag to override the global TLS enabled flag. + # enabled: true + cert: default + requireClientAuth: false + # -- Schema registry listeners. + schemaRegistry: + enabled: true + port: 8081 + kafkaEndpoint: default + # default is "http_basic" + authenticationMethod: + tls: + # Optional flag to override the global TLS enabled flag. + # enabled: true + cert: default + requireClientAuth: false + external: + default: + # enabled: true + port: 8084 + advertisedPorts: + - 30081 + tls: + # enabled: true + cert: external + requireClientAuth: false + # default is "http_basic" + authenticationMethod: + # -- HTTP API listeners (aka PandaProxy). + http: + enabled: true + port: 8082 + kafkaEndpoint: default + # default is "http_basic" + authenticationMethod: + tls: + # Optional flag to override the global TLS enabled flag. + # enabled: true + cert: default + requireClientAuth: false + external: + default: + # enabled: true + port: 8083 + # prefixTemplate: "" + advertisedPorts: + - 30082 + tls: + # enabled: true + cert: external + requireClientAuth: false + # default is "http_basic" + authenticationMethod: + +# Expert Config +# Here be dragons! +# +# -- This section contains various settings supported by Redpanda that may not work +# correctly in a Kubernetes cluster. Changing these settings comes with some risk. +# +# Use these settings to customize various Redpanda configurations that are not covered in other sections. +# These values have no impact on the configuration or behavior of the Kubernetes objects deployed by Helm, +# and therefore should not be modified for the purpose of configuring those objects. +# Instead, these settings get passed directly to the Redpanda binary at startup. +# For descriptions of these properties, +# see the [configuration documentation](https://docs.redpanda.com/docs/cluster-administration/configuration/). +config: + rpk: {} + # additional_start_flags: # List of flags to pass to rpk, e.g., ` "--idle-poll-time-us=0"` + # -- [Cluster Configuration Properties](https://docs.redpanda.com/current/reference/properties/cluster-properties/) + cluster: {} + + # -- Tunable cluster properties. + # Deprecated: all settings here may be specified via `config.cluster`. + tunable: + # -- See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#log_segment_size_min). + log_segment_size_min: 16777216 # 16 mb + # -- See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#log_segment_size_max). + log_segment_size_max: 268435456 # 256 mb + # -- See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#compacted_log_segment_size). + compacted_log_segment_size: 67108864 # 64 mb + # -- See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#max_compacted_log_segment_size). + max_compacted_log_segment_size: 536870912 # 512 mb + # -- See the [property reference documentation](https://docs.redpanda.com/docs/reference/cluster-properties/#kafka_connection_rate_limit). + kafka_connection_rate_limit: 1000 + + # -- [Broker (node) Configuration Properties](https://docs.redpanda.com/docs/reference/broker-properties/). + node: + # -- Crash loop limit + # A limit on the number of consecutive times a broker can crash within one hour before its crash-tracking logic is reset. + # This limit prevents a broker from getting stuck in an infinite cycle of crashes. + # User can disable this crash loop limit check by the following action: + # + # * One hour elapses since the last crash + # * The node configuration file, redpanda.yaml, is updated via config.cluster or config.node or config.tunable objects + # * The startup_log file in the node’s data_directory is manually deleted + # + # Default to 5 + # REF: https://docs.redpanda.com/current/reference/broker-properties/#crash_loop_limit + crash_loop_limit: 5 + + # Reference schema registry client https://docs.redpanda.com/current/reference/node-configuration-sample/ + schema_registry_client: {} + # # Number of times to retry a request to a broker + # # Default: 5 + # retries: 5 + # + # # Delay (in milliseconds) for initial retry backoff + # # Default: 100ms + # retry_base_backoff_ms: 100 + # + # # Number of records to batch before sending to broker + # # Default: 1000 + # produce_batch_record_count: 1000 + # + # # Number of bytes to batch before sending to broker + # # Defautl 1MiB + # produce_batch_size_bytes: 1048576 + # + # # Delay (in milliseconds) to wait before sending batch + # # Default: 100ms + # produce_batch_delay_ms: 100 + # + # # Interval (in milliseconds) for consumer request timeout + # # Default: 100ms + # consumer_request_timeout_ms: 100 + # + # # Max bytes to fetch per request + # # Default: 1MiB + # consumer_request_max_bytes: 1048576 + # + # # Timeout (in milliseconds) for consumer session + # # Default: 10s + # consumer_session_timeout_ms: 10000 + # + # # Timeout (in milliseconds) for consumer rebalance + # # Default: 2s + # consumer_rebalance_timeout_ms: 2000 + # + # # Interval (in milliseconds) for consumer heartbeats + # # Default: 500ms + # consumer_heartbeat_interval_ms: 500 + + # Reference panda proxy client https://docs.redpanda.com/current/reference/node-configuration-sample/ + pandaproxy_client: {} + # # Number of times to retry a request to a broker + # # Default: 5 + # retries: 5 + # + # # Delay (in milliseconds) for initial retry backoff + # # Default: 100ms + # retry_base_backoff_ms: 100 + # + # # Number of records to batch before sending to broker + # # Default: 1000 + # produce_batch_record_count: 1000 + # + # # Number of bytes to batch before sending to broker + # # Defautl 1MiB + # produce_batch_size_bytes: 1048576 + # + # # Delay (in milliseconds) to wait before sending batch + # # Default: 100ms + # produce_batch_delay_ms: 100 + # + # # Interval (in milliseconds) for consumer request timeout + # # Default: 100ms + # consumer_request_timeout_ms: 100 + # + # # Max bytes to fetch per request + # # Default: 1MiB + # consumer_request_max_bytes: 1048576 + # + # # Timeout (in milliseconds) for consumer session + # # Default: 10s + # consumer_session_timeout_ms: 10000 + # + # # Timeout (in milliseconds) for consumer rebalance + # # Default: 2s + # consumer_rebalance_timeout_ms: 2000 + # + # # Interval (in milliseconds) for consumer heartbeats + # # Default: 500ms + # consumer_heartbeat_interval_ms: 500 + + # Invalid properties + # Any of these properties will be ignored. These otherwise valid properties are not allowed + # to be used in this section since they impact deploying Redpanda in Kubernetes. + # Make use of the above sections to modify these values instead (see comments below). + # admin: "127.0.0.1:9644" # Address and port of admin server: use listeners.admin + # admin_api_tls: validate_many # TLS configuration for admin HTTP server: use listeners.admin.tls + # advertised_kafka_api: None # Address of Kafka API published to the clients + # advertised_pandaproxy_api: None # Rest API address and port to publish to client + # advertised_rpc_api: None # Address of RPC endpoint published to other cluster members + # enable_admin_api: true # Enable the admin API + # enable_sasl: false # Enable SASL authentication for Kafka connections + # kafka_api: "127.0.0.1:9092" # Address and port of an interface to listen for Kafka API requests + # kafka_api_tls: None # TLS configuration for Kafka API endpoint + # pandaproxy_api: "0.0.0.0:8082" # Rest API listen address and port + # pandaproxy_api_tls: validate_many # TLS configuration for Pandaproxy api + # rpc_server: "127.0.0.1:33145" # IP address and port for RPC server + # rpc_server_tls: validate # TLS configuration for RPC server + # superusers: None # List of superuser usernames + +tests: + enabled: true diff --git a/index.yaml b/index.yaml index 3428d3e0f1..11a27801c8 100644 --- a/index.yaml +++ b/index.yaml @@ -16331,6 +16331,49 @@ entries: - assets/hpe/hpe-csi-info-metrics-1.0.2.tgz version: 1.0.2 instana-agent: + - annotations: + artifacthub.io/links: | + - name: Instana website + url: https://www.ibm.com/products/instana + - name: Instana Helm charts + url: https://github.com/instana/helm-charts + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Instana Agent + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: instana-agent + apiVersion: v2 + appVersion: 1.285.0 + created: "2024-11-15T00:01:46.463345776Z" + description: Instana Agent for Kubernetes + digest: bdab6ecb5f826e20f6d0f8e4560f6f4c87e46daf18bfc5ba12736060c7c59e92 + home: https://www.instana.com/ + icon: file://assets/icons/instana-agent.png + kubeVersion: '>=1.21-0' + maintainers: + - email: felix.marx@ibm.com + name: FelixMarxIBM + - email: henning.treu@ibm.com + name: htreu + - email: konrad.ohms@de.ibm.com + name: Konrad-Ohms + - email: fredrik.gundersen@ibm.com + name: FredrikAtIBM + - email: jefiyamj@ibm.com + name: Jefiya-MJ + - email: milica.cvrkota@ibm.com + name: Milica-Cvrkota-IBM + - email: Nagaraj.Kandoor@ibm.com + name: nagaraj-kandoor + - email: Vineeth.Soman@ibm.com + name: vineethsoman03 + - email: Rashmi.Swamy@ibm.com + name: rashmiswamyibm + name: instana-agent + sources: + - https://github.com/instana/instana-agent-docker + urls: + - assets/instana/instana-agent-2.0.2.tgz + version: 2.0.2 - annotations: artifacthub.io/links: | - name: Instana website @@ -25428,6 +25471,38 @@ entries: catalog.cattle.io/kube-version: '>=1.22.0-0' catalog.cattle.io/release-name: linkerd-control-plane apiVersion: v2 + appVersion: edge-24.11.3 + created: "2024-11-15T00:01:45.328647595Z" + dependencies: + - name: partials + repository: file://../partials + version: 0.1.0 + description: 'Linkerd gives you observability, reliability, and security for your + microservices — with no code change required. ' + digest: 2380d98d90e83817a35c918c02398a48bb06caecb610d5008973dc4efb5f5324 + home: https://linkerd.io + icon: file://assets/icons/linkerd-control-plane.png + keywords: + - service-mesh + kubeVersion: '>=1.22.0-0' + maintainers: + - email: cncf-linkerd-dev@lists.cncf.io + name: Linkerd authors + url: https://linkerd.io/ + name: linkerd-control-plane + sources: + - https://github.com/linkerd/linkerd2/ + type: application + urls: + - assets/buoyant/linkerd-control-plane-2024.11.3.tgz + version: 2024.11.3 + - annotations: + catalog.cattle.io/auto-install: linkerd-crds + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Linkerd Control Plane + catalog.cattle.io/kube-version: '>=1.22.0-0' + catalog.cattle.io/release-name: linkerd-control-plane + apiVersion: v2 appVersion: edge-24.11.2 created: "2024-11-10T00:37:52.497361594Z" dependencies: @@ -25436,7 +25511,7 @@ entries: version: 0.1.0 description: 'Linkerd gives you observability, reliability, and security for your microservices — with no code change required. ' - digest: 44f58cb2fe1beb3b7546faeab2bb56d0ef4ad6a70e88cd6934f9a37b155192a6 + digest: 1456d6cc28d1255b85b8d8b90442964abb1cda874bdb111b7755bd82edaf05dc home: https://linkerd.io icon: file://assets/icons/linkerd-control-plane.png keywords: @@ -26851,6 +26926,36 @@ entries: - assets/buoyant/linkerd-control-plane-1.12.5.tgz version: 1.12.5 linkerd-crds: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Linkerd CRDs + catalog.cattle.io/kube-version: '>=1.22.0-0' + catalog.cattle.io/release-name: linkerd-crds + apiVersion: v2 + created: "2024-11-15T00:01:45.410044085Z" + dependencies: + - name: partials + repository: file://../partials + version: 0.1.0 + description: 'Linkerd gives you observability, reliability, and security for your + microservices — with no code change required. ' + digest: 83d4ea7da0aef4851eaa918f3b8c358d76830b6501cace0b0fc3755aeff741a7 + home: https://linkerd.io + icon: file://assets/icons/linkerd-crds.png + keywords: + - service-mesh + kubeVersion: '>=1.22.0-0' + maintainers: + - email: cncf-linkerd-dev@lists.cncf.io + name: Linkerd authors + url: https://linkerd.io/ + name: linkerd-crds + sources: + - https://github.com/linkerd/linkerd2/ + type: application + urls: + - assets/buoyant/linkerd-crds-2024.11.3.tgz + version: 2024.11.3 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: Linkerd CRDs @@ -35851,6 +35956,29 @@ entries: - assets/fairwinds/polaris-5.11.1.tgz version: 5.11.1 psmdb-db: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Percona Server for MongoDB + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: psmdb-db + apiVersion: v2 + appVersion: 1.18.0 + created: "2024-11-15T00:01:49.641901664Z" + description: A Helm chart for installing Percona Server MongoDB Cluster Databases + using the PSMDB Operator. + digest: b1701cfaf70d98f35a67b88bdbd0b984d435dadf48af318b4097c440f6e98cc8 + home: https://www.percona.com/doc/kubernetes-operator-for-psmongodb/index.html + icon: file://assets/icons/psmdb-db.png + kubeVersion: '>=1.21-0' + maintainers: + - email: tomislav.plavcic@percona.com + name: tplavcic + - email: natalia.marukovich@percona.com + name: nmarukovich + name: psmdb-db + urls: + - assets/percona/psmdb-db-1.18.0.tgz + version: 1.18.0 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: Percona Server for MongoDB @@ -36098,6 +36226,30 @@ entries: - assets/percona/psmdb-db-1.14.4.tgz version: 1.14.4 psmdb-operator: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Percona Operator for MongoDB + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: psmdb-operator + apiVersion: v2 + appVersion: 1.18.0 + created: "2024-11-15T00:01:49.667519503Z" + description: A Helm chart for deploying the Percona Operator for MongoDB + digest: 934a6611de3255235015792c20ec9cf60affb645570b957d486e95b74fad6aac + home: https://docs.percona.com/percona-operator-for-mongodb/ + icon: file://assets/icons/psmdb-operator.png + kubeVersion: '>=1.21-0' + maintainers: + - email: tomislav.plavcic@percona.com + name: tplavcic + - email: natalia.marukovich@percona.com + name: nmarukovich + - email: sergey.pronin@percona.com + name: spron-in + name: psmdb-operator + urls: + - assets/percona/psmdb-operator-1.18.0.tgz + version: 1.18.0 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: Percona Operator for MongoDB @@ -37064,6 +37216,48 @@ entries: - assets/quobyte/quobyte-cluster-0.1.8.tgz version: 0.1.8 redpanda: + - annotations: + artifacthub.io/images: | + - name: redpanda + image: docker.redpanda.com/redpandadata/redpanda:v24.2.7 + - name: busybox + image: busybox:latest + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Documentation + url: https://docs.redpanda.com + - name: "Helm (>= 3.10.0)" + url: https://helm.sh/docs/intro/install/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Redpanda + catalog.cattle.io/kube-version: '>=1.21-0' + catalog.cattle.io/release-name: redpanda + apiVersion: v2 + appVersion: v24.2.7 + created: "2024-11-15T00:01:50.070399337Z" + dependencies: + - condition: console.enabled + name: console + repository: https://charts.redpanda.com + version: '>=0.5 <1.0' + - condition: connectors.enabled + name: connectors + repository: https://charts.redpanda.com + version: '>=0.1.2 <1.0' + description: Redpanda is the real-time engine for modern apps. + digest: 4f9ddd06bcd789b2838157578c13903ebf7406aa7f6a4209b5fb64ccb8513c92 + icon: file://assets/icons/redpanda.svg + kubeVersion: '>=1.21-0' + maintainers: + - name: redpanda-data + url: https://github.com/orgs/redpanda-data/people + name: redpanda + sources: + - https://github.com/redpanda-data/helm-charts + type: application + urls: + - assets/redpanda/redpanda-5.9.10.tgz + version: 5.9.10 - annotations: artifacthub.io/images: | - name: redpanda @@ -48148,4 +48342,4 @@ entries: urls: - assets/netfoundry/ziti-host-1.5.1.tgz version: 1.5.1 -generated: "2024-11-14T00:01:44.600510043Z" +generated: "2024-11-15T00:01:45.043320849Z"