From 24ee7c82810959e638d3f4bed890bb9044d42c4e Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" Date: Sat, 16 Nov 2024 00:07:05 +0000 Subject: [PATCH] Added chart versions: airlock/microgateway: - 4.4.1 airlock/microgateway-cni: - 4.4.1 kasten/k10: - 7.0.14 kuma/kuma: - 2.9.1 netscaler/netscaler-cpx-with-ingress-controller: - 2.2.10 netscaler/netscaler-ingress-controller: - 2.2.10 trilio/k8s-triliovault-operator: - 5.0.0 --- assets/airlock/microgateway-4.4.1.tgz | Bin 0 -> 82234 bytes assets/airlock/microgateway-cni-4.4.1.tgz | Bin 0 -> 10942 bytes assets/kasten/k10-7.0.1401.tgz | Bin 0 -> 230732 bytes assets/kuma/kuma-2.9.1.tgz | Bin 0 -> 75599 bytes ...ler-cpx-with-ingress-controller-2.2.10.tgz | Bin 0 -> 39789 bytes .../netscaler-ingress-controller-2.2.10.tgz | Bin 0 -> 36318 bytes .../trilio/k8s-triliovault-operator-5.0.0.tgz | Bin 0 -> 109779 bytes .../microgateway-cni/4.4.1/.helmignore | 27 + .../airlock/microgateway-cni/4.4.1/Chart.yaml | 43 + .../airlock/microgateway-cni/4.4.1/README.md | 137 + .../microgateway-cni/4.4.1/gke-values.yaml | 4 + .../4.4.1/openshift-values.yaml | 15 + .../microgateway-cni/4.4.1/questions.yml | 18 + .../4.4.1/templates/NOTES.txt | 15 + .../4.4.1/templates/_helpers.tpl | 101 + .../4.4.1/templates/clusterrole.yaml | 22 + .../4.4.1/templates/clusterrolebinding.yaml | 20 + .../4.4.1/templates/configmap.yaml | 22 + .../4.4.1/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../4.4.1/templates/scc-role.yaml | 22 + .../4.4.1/templates/scc-rolebinding.yaml | 20 + .../4.4.1/templates/serviceaccount.yaml | 13 + .../4.4.1/templates/tests/rbac.yaml | 64 + .../4.4.1/templates/tests/test-install.yaml | 103 + .../microgateway-cni/4.4.1/values.schema.json | 225 + .../microgateway-cni/4.4.1/values.yaml | 85 + charts/airlock/microgateway/4.2.3/.helmignore | 3 +- charts/airlock/microgateway/4.2.3/Chart.yaml | 15 +- charts/airlock/microgateway/4.2.3/README.md | 134 +- ...cesscontrols.microgateway.airlock.com.yaml | 124 - ...ntsecurities.microgateway.airlock.com.yaml | 127 - .../denyrules.microgateway.airlock.com.yaml | 1508 ---- ...nvoyclusters.microgateway.airlock.com.yaml | 58 - ...nfigurations.microgateway.airlock.com.yaml | 182 - ...aderrewrites.microgateway.airlock.com.yaml | 759 -- ...propagations.microgateway.airlock.com.yaml | 108 - .../crds/limits.microgateway.airlock.com.yaml | 453 -- ...idcproviders.microgateway.airlock.com.yaml | 301 - ...lyingparties.microgateway.airlock.com.yaml | 219 - .../openapis.microgateway.airlock.com.yaml | 167 - .../parsers.microgateway.airlock.com.yaml | 358 - ...ecargateways.microgateway.airlock.com.yaml | 731 -- .../telemetries.microgateway.airlock.com.yaml | 81 - .../microgateway/4.2.3/gke-values.yaml | 4 + .../microgateway/4.2.3/openshift-values.yaml | 15 + .../airlock/microgateway/4.2.3/questions.yml | 18 + .../microgateway/4.2.3/templates/NOTES.txt | 23 +- .../microgateway/4.2.3/templates/_helpers.tpl | 101 +- .../4.2.3/templates/clusterrole.yaml | 22 + .../4.2.3/templates/clusterrolebinding.yaml | 20 + .../4.2.3/templates/configmap.yaml | 22 + .../4.2.3/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../templates/operator/_webhooks.gen.tpl | 299 - .../4.2.3/templates/operator/configmap.yaml | 322 - .../4.2.3/templates/operator/deployment.yaml | 138 - .../operator/manager-clusterrole.yaml | 14 - .../operator/manager-clusterrolebinding.yaml | 20 - .../templates/operator/mutating-webhook.yaml | 14 - .../4.2.3/templates/operator/podmonitor.yaml | 26 - .../operator/validating-webhook.yaml | 14 - .../4.2.3/templates/scc-role.yaml | 22 + .../4.2.3/templates/scc-rolebinding.yaml | 20 + .../4.2.3/templates/serviceaccount.yaml | 13 + .../4.2.3/templates/tests/rbac.yaml | 105 +- .../4.2.3/templates/tests/test-install.yaml | 229 +- .../microgateway/4.2.3/values.schema.json | 321 +- charts/airlock/microgateway/4.2.3/values.yaml | 188 +- charts/airlock/microgateway/4.3.0/.helmignore | 3 +- charts/airlock/microgateway/4.3.0/Chart.yaml | 15 +- charts/airlock/microgateway/4.3.0/README.md | 147 +- .../airlock/microgateway/4.3.0/app-readme.md | 28 - ...cesscontrols.microgateway.airlock.com.yaml | 124 - ...nvoyclusters.microgateway.airlock.com.yaml | 58 - ...nfigurations.microgateway.airlock.com.yaml | 185 - ...yhttpfilters.microgateway.airlock.com.yaml | 58 - .../graphqls.microgateway.airlock.com.yaml | 88 - ...aderrewrites.microgateway.airlock.com.yaml | 759 -- .../parsers.microgateway.airlock.com.yaml | 358 - ...ecargateways.microgateway.airlock.com.yaml | 758 -- .../4.3.0/dashboards/license.json | 521 -- .../4.3.0/dashboards/overview.json | 1138 --- .../microgateway/4.3.0/gke-values.yaml | 4 + .../microgateway/4.3.0/openshift-values.yaml | 15 + .../airlock/microgateway/4.3.0/questions.yml | 18 + .../microgateway/4.3.0/templates/NOTES.txt | 35 +- .../microgateway/4.3.0/templates/_helpers.tpl | 122 +- .../4.3.0/templates/clusterrole.yaml | 22 + .../4.3.0/templates/clusterrolebinding.yaml | 20 + .../4.3.0/templates/configmap.yaml | 22 + .../4.3.0/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../templates/operator/_operator_helpers.tpl | 42 - .../4.3.0/templates/operator/_rbac.gen.tpl | 237 - .../templates/operator/_webhooks.gen.tpl | 339 - .../4.3.0/templates/operator/configmap.yaml | 394 - .../templates/operator/metrics-service.yaml | 47 - .../4.3.0/templates/operator/role.yaml | 45 - .../4.3.0/templates/operator/rolebinding.yaml | 20 - .../templates/operator/selfsigned-issuer.yaml | 13 - .../templates/operator/serviceaccount.yaml | 13 - .../templates/operator/servicemonitor.yaml | 60 - .../operator/serving-certificate.yaml | 19 - .../templates/operator/webhook-service.yaml | 23 - .../4.3.0/templates/operator/xds-service.yaml | 24 - .../4.3.0/templates/scc-role.yaml | 22 + .../4.3.0/templates/scc-rolebinding.yaml | 20 + .../4.3.0/templates/serviceaccount.yaml | 13 + .../4.3.0/templates/tests/rbac.yaml | 141 +- .../4.3.0/templates/tests/service.yaml | 23 - .../4.3.0/templates/tests/statefulset.yaml | 56 - .../4.3.0/templates/tests/test-install.yaml | 256 +- .../microgateway/4.3.0/values.schema.json | 547 +- charts/airlock/microgateway/4.3.0/values.yaml | 266 +- charts/airlock/microgateway/4.3.1/.helmignore | 3 +- charts/airlock/microgateway/4.3.1/Chart.yaml | 15 +- charts/airlock/microgateway/4.3.1/README.md | 147 +- .../airlock/microgateway/4.3.1/app-readme.md | 28 - ...cesscontrols.microgateway.airlock.com.yaml | 124 - ...ntsecurities.microgateway.airlock.com.yaml | 139 - .../denyrules.microgateway.airlock.com.yaml | 1804 ----- ...yhttpfilters.microgateway.airlock.com.yaml | 58 - .../graphqls.microgateway.airlock.com.yaml | 88 - ...aderrewrites.microgateway.airlock.com.yaml | 759 -- ...propagations.microgateway.airlock.com.yaml | 108 - .../crds/limits.microgateway.airlock.com.yaml | 651 -- ...idcproviders.microgateway.airlock.com.yaml | 305 - ...lyingparties.microgateway.airlock.com.yaml | 224 - .../openapis.microgateway.airlock.com.yaml | 167 - .../parsers.microgateway.airlock.com.yaml | 358 - ...disproviders.microgateway.airlock.com.yaml | 159 - ...ionhandlings.microgateway.airlock.com.yaml | 77 - .../telemetries.microgateway.airlock.com.yaml | 96 - .../4.3.1/dashboards/license.json | 521 -- .../4.3.1/dashboards/overview.json | 1138 --- .../microgateway/4.3.1/gke-values.yaml | 4 + .../microgateway/4.3.1/openshift-values.yaml | 15 + .../airlock/microgateway/4.3.1/questions.yml | 18 + .../microgateway/4.3.1/templates/NOTES.txt | 35 +- .../microgateway/4.3.1/templates/_helpers.tpl | 122 +- .../4.3.1/templates/clusterrole.yaml | 22 + .../4.3.1/templates/clusterrolebinding.yaml | 20 + .../4.3.1/templates/configmap.yaml | 22 + .../4.3.1/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../templates/operator/_operator_helpers.tpl | 42 - .../4.3.1/templates/operator/_rbac.gen.tpl | 237 - .../operator/dashboard-configmap.yaml | 28 - .../4.3.1/templates/operator/deployment.yaml | 143 - .../templates/operator/manager-role.yaml | 33 - .../operator/manager-rolebinding.yaml | 45 - .../templates/operator/metrics-service.yaml | 47 - .../templates/operator/mutating-webhook.yaml | 28 - .../4.3.1/templates/operator/podmonitor.yaml | 27 - .../4.3.1/templates/operator/role.yaml | 45 - .../4.3.1/templates/operator/rolebinding.yaml | 20 - .../templates/operator/selfsigned-issuer.yaml | 13 - .../templates/operator/serviceaccount.yaml | 13 - .../templates/operator/servicemonitor.yaml | 60 - .../operator/serving-certificate.yaml | 19 - .../operator/validating-webhook.yaml | 28 - .../templates/operator/webhook-service.yaml | 23 - .../4.3.1/templates/operator/xds-service.yaml | 24 - .../4.3.1/templates/scc-role.yaml | 22 + .../4.3.1/templates/scc-rolebinding.yaml | 20 + .../4.3.1/templates/serviceaccount.yaml | 13 + .../4.3.1/templates/tests/rbac.yaml | 141 +- .../4.3.1/templates/tests/service.yaml | 23 - .../4.3.1/templates/tests/statefulset.yaml | 56 - .../4.3.1/templates/tests/test-install.yaml | 256 +- .../microgateway/4.3.1/values.schema.json | 547 +- charts/airlock/microgateway/4.3.1/values.yaml | 266 +- charts/airlock/microgateway/4.3.3/.helmignore | 3 +- charts/airlock/microgateway/4.3.3/Chart.yaml | 15 +- charts/airlock/microgateway/4.3.3/README.md | 147 +- .../airlock/microgateway/4.3.3/app-readme.md | 28 - ...cesscontrols.microgateway.airlock.com.yaml | 124 - ...ntsecurities.microgateway.airlock.com.yaml | 139 - .../denyrules.microgateway.airlock.com.yaml | 1804 ----- ...nvoyclusters.microgateway.airlock.com.yaml | 58 - ...nfigurations.microgateway.airlock.com.yaml | 185 - ...yhttpfilters.microgateway.airlock.com.yaml | 58 - ...aderrewrites.microgateway.airlock.com.yaml | 759 -- ...propagations.microgateway.airlock.com.yaml | 108 - .../crds/limits.microgateway.airlock.com.yaml | 651 -- ...idcproviders.microgateway.airlock.com.yaml | 305 - ...lyingparties.microgateway.airlock.com.yaml | 224 - .../openapis.microgateway.airlock.com.yaml | 167 - ...disproviders.microgateway.airlock.com.yaml | 159 - ...ionhandlings.microgateway.airlock.com.yaml | 77 - ...ecargateways.microgateway.airlock.com.yaml | 758 -- .../telemetries.microgateway.airlock.com.yaml | 96 - .../4.3.3/dashboards/blockMetrics.json | 758 -- .../4.3.3/dashboards/license.json | 521 -- .../microgateway/4.3.3/gke-values.yaml | 4 + .../microgateway/4.3.3/openshift-values.yaml | 15 + .../airlock/microgateway/4.3.3/questions.yml | 18 + .../microgateway/4.3.3/templates/NOTES.txt | 48 +- .../microgateway/4.3.3/templates/_helpers.tpl | 122 +- .../4.3.3/templates/clusterrole.yaml | 22 + .../4.3.3/templates/clusterrolebinding.yaml | 20 + .../4.3.3/templates/configmap.yaml | 22 + .../4.3.3/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../templates/operator/_operator_helpers.tpl | 42 - .../4.3.3/templates/operator/_rbac.gen.tpl | 237 - .../templates/operator/_webhooks.gen.tpl | 339 - .../4.3.3/templates/operator/configmap.yaml | 394 - .../operator/dashboard-configmap.yaml | 28 - .../4.3.3/templates/operator/deployment.yaml | 143 - .../templates/operator/manager-role.yaml | 33 - .../operator/manager-rolebinding.yaml | 45 - .../templates/operator/metrics-service.yaml | 47 - .../templates/operator/mutating-webhook.yaml | 28 - .../4.3.3/templates/operator/podmonitor.yaml | 27 - .../4.3.3/templates/operator/role.yaml | 45 - .../4.3.3/templates/operator/rolebinding.yaml | 20 - .../templates/operator/selfsigned-issuer.yaml | 13 - .../templates/operator/serviceaccount.yaml | 13 - .../templates/operator/servicemonitor.yaml | 60 - .../operator/serving-certificate.yaml | 19 - .../operator/validating-webhook.yaml | 28 - .../templates/operator/webhook-service.yaml | 23 - .../4.3.3/templates/operator/xds-service.yaml | 24 - .../4.3.3/templates/scc-role.yaml | 22 + .../4.3.3/templates/scc-rolebinding.yaml | 20 + .../4.3.3/templates/serviceaccount.yaml | 13 + .../4.3.3/templates/tests/rbac.yaml | 141 +- .../4.3.3/templates/tests/service.yaml | 23 - .../4.3.3/templates/tests/statefulset.yaml | 56 - .../4.3.3/templates/tests/test-install.yaml | 256 +- .../microgateway/4.3.3/values.schema.json | 547 +- charts/airlock/microgateway/4.3.3/values.yaml | 266 +- charts/airlock/microgateway/4.3.4/.helmignore | 3 +- charts/airlock/microgateway/4.3.4/Chart.yaml | 15 +- charts/airlock/microgateway/4.3.4/README.md | 147 +- .../airlock/microgateway/4.3.4/app-readme.md | 28 - ...cesscontrols.microgateway.airlock.com.yaml | 124 - ...ntsecurities.microgateway.airlock.com.yaml | 139 - .../denyrules.microgateway.airlock.com.yaml | 1804 ----- ...nvoyclusters.microgateway.airlock.com.yaml | 58 - ...nfigurations.microgateway.airlock.com.yaml | 185 - ...yhttpfilters.microgateway.airlock.com.yaml | 58 - .../graphqls.microgateway.airlock.com.yaml | 88 - ...aderrewrites.microgateway.airlock.com.yaml | 759 -- ...propagations.microgateway.airlock.com.yaml | 108 - .../crds/limits.microgateway.airlock.com.yaml | 651 -- ...idcproviders.microgateway.airlock.com.yaml | 305 - ...lyingparties.microgateway.airlock.com.yaml | 224 - .../openapis.microgateway.airlock.com.yaml | 167 - .../parsers.microgateway.airlock.com.yaml | 358 - ...disproviders.microgateway.airlock.com.yaml | 159 - ...ionhandlings.microgateway.airlock.com.yaml | 77 - ...ecargateways.microgateway.airlock.com.yaml | 758 -- .../telemetries.microgateway.airlock.com.yaml | 96 - .../4.3.4/dashboards/blockLogs.json | 510 -- .../4.3.4/dashboards/blockMetrics.json | 758 -- .../4.3.4/dashboards/license.json | 521 -- .../microgateway/4.3.4/gke-values.yaml | 4 + .../microgateway/4.3.4/openshift-values.yaml | 15 + .../airlock/microgateway/4.3.4/questions.yml | 18 + .../microgateway/4.3.4/templates/NOTES.txt | 48 +- .../microgateway/4.3.4/templates/_helpers.tpl | 122 +- .../4.3.4/templates/clusterrole.yaml | 22 + .../4.3.4/templates/clusterrolebinding.yaml | 20 + .../4.3.4/templates/configmap.yaml | 22 + .../4.3.4/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../templates/operator/_operator_helpers.tpl | 42 - .../4.3.4/templates/operator/_rbac.gen.tpl | 237 - .../templates/operator/_webhooks.gen.tpl | 339 - .../4.3.4/templates/operator/configmap.yaml | 394 - .../operator/dashboard-configmap.yaml | 28 - .../4.3.4/templates/operator/deployment.yaml | 143 - .../templates/operator/manager-role.yaml | 33 - .../operator/manager-rolebinding.yaml | 45 - .../templates/operator/metrics-service.yaml | 47 - .../templates/operator/mutating-webhook.yaml | 28 - .../4.3.4/templates/operator/podmonitor.yaml | 27 - .../4.3.4/templates/operator/role.yaml | 45 - .../4.3.4/templates/operator/rolebinding.yaml | 20 - .../templates/operator/selfsigned-issuer.yaml | 13 - .../templates/operator/serviceaccount.yaml | 13 - .../templates/operator/servicemonitor.yaml | 60 - .../operator/serving-certificate.yaml | 19 - .../operator/validating-webhook.yaml | 28 - .../templates/operator/webhook-service.yaml | 23 - .../4.3.4/templates/operator/xds-service.yaml | 24 - .../4.3.4/templates/scc-role.yaml | 22 + .../4.3.4/templates/scc-rolebinding.yaml | 20 + .../4.3.4/templates/serviceaccount.yaml | 13 + .../4.3.4/templates/tests/rbac.yaml | 141 +- .../4.3.4/templates/tests/service.yaml | 23 - .../4.3.4/templates/tests/statefulset.yaml | 56 - .../4.3.4/templates/tests/test-install.yaml | 256 +- .../microgateway/4.3.4/values.schema.json | 547 +- charts/airlock/microgateway/4.3.4/values.yaml | 266 +- charts/airlock/microgateway/4.4.1/.helmignore | 28 + charts/airlock/microgateway/4.4.1/Chart.yaml | 44 + charts/airlock/microgateway/4.4.1/README.md | 186 + .../{4.2.3 => 4.4.1}/app-readme.md | 0 ...cesscontrols.microgateway.airlock.com.yaml | 501 ++ ...ntsecurities.microgateway.airlock.com.yaml | 4 +- ...ritypolicies.microgateway.airlock.com.yaml | 476 ++ .../denyrules.microgateway.airlock.com.yaml | 16 +- ...nvoyclusters.microgateway.airlock.com.yaml | 4 +- ...nfigurations.microgateway.airlock.com.yaml | 4 +- ...yhttpfilters.microgateway.airlock.com.yaml | 4 +- .../graphqls.microgateway.airlock.com.yaml | 4 +- ...aderrewrites.microgateway.airlock.com.yaml | 2083 ++++++ ...propagations.microgateway.airlock.com.yaml | 51 +- .../crds/jwks.microgateway.airlock.com.yaml | 294 + .../crds/limits.microgateway.airlock.com.yaml | 4 +- ...idcproviders.microgateway.airlock.com.yaml | 67 +- ...lyingparties.microgateway.airlock.com.yaml | 47 +- .../openapis.microgateway.airlock.com.yaml | 4 +- .../parsers.microgateway.airlock.com.yaml | 4 +- ...disproviders.microgateway.airlock.com.yaml | 77 +- ...ionhandlings.microgateway.airlock.com.yaml | 34 +- ...ecargateways.microgateway.airlock.com.yaml | 4 +- .../telemetries.microgateway.airlock.com.yaml | 4 +- .../dashboards/blockLogs.json | 109 +- .../dashboards/blockMetrics.json | 41 +- .../dashboards/headerLogs.json} | 186 +- .../dashboards/license.json} | 1283 ++-- .../dashboards/logOnlyLogs.json} | 184 +- .../dashboards/logOnlyMetrics.json} | 213 +- .../{4.3.3 => 4.4.1}/dashboards/overview.json | 20 +- .../microgateway/4.4.1/templates/NOTES.txt | 61 + .../microgateway/4.4.1/templates/_helpers.tpl | 153 + .../templates/operator/_operator_helpers.tpl | 0 .../templates/operator/_rbac.gen.tpl | 197 +- .../templates/operator/_webhooks.gen.tpl | 60 + .../templates/operator/configmap.yaml | 15 +- .../operator/dashboard-configmap.yaml | 0 .../templates/operator/deployment.yaml | 0 .../templates/operator/manager-role.yaml | 0 .../operator/manager-rolebinding.yaml | 0 .../templates/operator/metrics-service.yaml | 0 .../templates/operator/mutating-webhook.yaml | 0 .../templates/operator/podmonitor.yaml | 0 .../templates/operator/role.yaml | 0 .../templates/operator/rolebinding.yaml | 0 .../templates/operator/selfsigned-issuer.yaml | 0 .../templates/operator/serviceaccount.yaml | 0 .../templates/operator/servicemonitor.yaml | 0 .../operator/serving-certificate.yaml | 0 .../operator/validating-webhook.yaml | 0 .../templates/operator/webhook-service.yaml | 0 .../templates/operator/xds-service.yaml | 0 .../4.4.1/templates/tests/rbac.yaml | 143 + .../templates/tests/service.yaml | 0 .../templates/tests/statefulset.yaml | 0 .../4.4.1/templates/tests/test-install.yaml | 227 + .../microgateway/4.4.1/values.schema.json | 572 ++ charts/airlock/microgateway/4.4.1/values.yaml | 237 + charts/kasten/k10/7.0.1401/Chart.lock | 9 + charts/kasten/k10/7.0.1401/Chart.yaml | 25 + charts/kasten/k10/7.0.1401/README.md | 344 + charts/kasten/k10/7.0.1401/app-readme.md | 5 + .../k10/7.0.1401/charts/grafana/.helmignore | 23 + .../k10/7.0.1401/charts/grafana/Chart.yaml | 35 + .../k10/7.0.1401/charts/grafana/README.md | 783 ++ .../charts/grafana/ci/default-values.yaml | 1 + .../grafana/ci/with-affinity-values.yaml | 16 + .../ci/with-dashboard-json-values.yaml | 53 + .../grafana/ci/with-dashboard-values.yaml | 19 + .../ci/with-extraconfigmapmounts-values.yaml | 7 + .../ci/with-image-renderer-values.yaml | 107 + .../grafana/ci/with-nondefault-values.yaml | 6 + .../charts/grafana/ci/with-persistence.yaml | 3 + .../ci/with-sidecars-envvaluefrom-values.yaml | 38 + .../grafana/dashboards/custom-dashboard.json | 1 + .../charts/grafana/templates/NOTES.txt | 55 + .../charts/grafana/templates/_config.tpl | 176 + .../charts/grafana/templates/_helpers.tpl | 276 + .../charts/grafana/templates/_pod.tpl | 1329 ++++ .../charts/grafana/templates/clusterrole.yaml | 25 + .../grafana/templates/clusterrolebinding.yaml | 24 + .../grafana/templates/configSecret.yaml | 43 + .../configmap-dashboard-provider.yaml | 15 + .../charts/grafana/templates/configmap.yaml | 20 + .../templates/dashboards-json-configmap.yaml | 35 + .../charts/grafana/templates/deployment.yaml | 54 + .../grafana/templates/extra-manifests.yaml | 4 + .../grafana/templates/headless-service.yaml | 22 + .../charts/grafana/templates/hpa.yaml | 52 + .../templates/image-renderer-deployment.yaml | 200 + .../grafana/templates/image-renderer-hpa.yaml | 47 + .../image-renderer-network-policy.yaml | 79 + .../templates/image-renderer-service.yaml | 31 + .../image-renderer-servicemonitor.yaml | 48 + .../charts/grafana/templates/ingress.yaml | 78 + .../grafana/templates/networkpolicy.yaml | 61 + .../templates/poddisruptionbudget.yaml | 22 + .../grafana/templates/podsecuritypolicy.yaml | 49 + .../charts/grafana/templates/pvc.yaml | 39 + .../charts/grafana/templates/role.yaml | 32 + .../charts/grafana/templates/rolebinding.yaml | 25 + .../charts/grafana/templates/secret-env.yaml | 14 + .../charts/grafana/templates/secret.yaml | 16 + .../charts/grafana/templates/service.yaml | 67 + .../grafana/templates/serviceaccount.yaml | 17 + .../grafana/templates/servicemonitor.yaml | 52 + .../charts/grafana/templates/statefulset.yaml | 58 + .../templates/tests/test-configmap.yaml | 20 + .../tests/test-podsecuritypolicy.yaml | 32 + .../grafana/templates/tests/test-role.yaml | 17 + .../templates/tests/test-rolebinding.yaml | 20 + .../templates/tests/test-serviceaccount.yaml | 12 + .../charts/grafana/templates/tests/test.yaml | 53 + .../k10/7.0.1401/charts/grafana/values.yaml | 1392 ++++ .../7.0.1401/charts/prometheus/.helmignore | 23 + .../k10/7.0.1401/charts/prometheus/Chart.lock | 15 + .../k10/7.0.1401/charts/prometheus/Chart.yaml | 53 + .../k10/7.0.1401/charts/prometheus/OWNERS | 6 + .../k10/7.0.1401/charts/prometheus/README.md | 382 + .../charts/alertmanager/.helmignore | 25 + .../prometheus/charts/alertmanager/Chart.yaml | 24 + .../prometheus/charts/alertmanager/README.md | 62 + .../alertmanager/ci/config-reload-values.yaml | 2 + .../charts/alertmanager/templates/NOTES.txt | 21 + .../alertmanager/templates/_helpers.tpl | 92 + .../alertmanager/templates/configmap.yaml | 21 + .../alertmanager/templates/ingress.yaml | 44 + .../templates/ingressperreplica.yaml | 56 + .../charts/alertmanager/templates/pdb.yaml | 14 + .../templates/serviceaccount.yaml | 14 + .../templates/serviceperreplica.yaml | 44 + .../alertmanager/templates/services.yaml | 75 + .../alertmanager/templates/statefulset.yaml | 254 + .../templates/tests/test-connection.yaml | 20 + .../charts/alertmanager/templates/vpa.yaml | 26 + .../charts/alertmanager/values.schema.json | 946 +++ .../charts/alertmanager/values.yaml | 404 ++ .../charts/kube-state-metrics/.helmignore | 21 + .../charts/kube-state-metrics/Chart.yaml | 26 + .../charts/kube-state-metrics/README.md | 85 + .../kube-state-metrics/templates/NOTES.txt | 23 + .../kube-state-metrics/templates/_helpers.tpl | 156 + .../templates/ciliumnetworkpolicy.yaml | 33 + .../templates/clusterrolebinding.yaml | 20 + .../templates/crs-configmap.yaml | 16 + .../templates/deployment.yaml | 336 + .../templates/extra-manifests.yaml | 4 + .../templates/kubeconfig-secret.yaml | 12 + .../templates/networkpolicy.yaml | 43 + .../kube-state-metrics/templates/pdb.yaml | 18 + .../templates/podsecuritypolicy.yaml | 39 + .../templates/psp-clusterrole.yaml | 19 + .../templates/psp-clusterrolebinding.yaml | 16 + .../templates/rbac-configmap.yaml | 22 + .../kube-state-metrics/templates/role.yaml | 212 + .../templates/rolebinding.yaml | 24 + .../kube-state-metrics/templates/service.yaml | 53 + .../templates/serviceaccount.yaml | 18 + .../templates/servicemonitor.yaml | 120 + .../templates/stsdiscovery-role.yaml | 26 + .../templates/stsdiscovery-rolebinding.yaml | 17 + .../templates/verticalpodautoscaler.yaml | 44 + .../charts/kube-state-metrics/values.yaml | 542 ++ .../prometheus-node-exporter/.helmignore | 21 + .../prometheus-node-exporter/Chart.yaml | 25 + .../charts/prometheus-node-exporter/README.md | 96 + .../ci/common-labels-values.yaml | 4 + .../ci/default-values.yaml | 1 + .../ci/networkpolicy-values.yaml | 5 + .../ci/pod-labels-values.yaml | 4 + .../ci/port-values.yaml | 3 + .../ci/service-labels-values.yaml | 5 + .../ci/serviceport-values.yaml | 3 + .../templates/NOTES.txt | 29 + .../templates/_helpers.tpl | 202 + .../templates/clusterrole.yaml | 19 + .../templates/clusterrolebinding.yaml | 20 + .../templates/daemonset.yaml | 315 + .../templates/endpoints.yaml | 18 + .../templates/extra-manifests.yaml | 4 + .../templates/networkpolicy.yaml | 27 + .../templates/podmonitor.yaml | 91 + .../templates/psp-clusterrole.yaml | 14 + .../templates/psp-clusterrolebinding.yaml | 16 + .../templates/psp.yaml | 49 + .../templates/rbac-configmap.yaml | 16 + .../templates/service.yaml | 38 + .../templates/serviceaccount.yaml | 18 + .../templates/servicemonitor.yaml | 61 + .../templates/verticalpodautoscaler.yaml | 40 + .../prometheus-node-exporter/values.yaml | 566 ++ .../charts/prometheus-pushgateway/.helmignore | 24 + .../charts/prometheus-pushgateway/Chart.yaml | 24 + .../charts/prometheus-pushgateway/README.md | 88 + .../templates/NOTES.txt | 19 + .../templates/_helpers.tpl | 304 + .../templates/deployment.yaml | 29 + .../templates/extra-manifests.yaml | 8 + .../templates/ingress.yaml | 50 + .../templates/networkpolicy.yaml | 26 + .../prometheus-pushgateway/templates/pdb.yaml | 14 + .../templates/pushgateway-pvc.yaml | 29 + .../templates/secret.yaml | 10 + .../templates/service.yaml | 45 + .../templates/serviceaccount.yaml | 17 + .../templates/servicemonitor.yaml | 51 + .../templates/statefulset.yaml | 49 + .../charts/prometheus-pushgateway/values.yaml | 371 + .../charts/prometheus/templates/NOTES.txt | 113 + .../charts/prometheus/templates/_helpers.tpl | 238 + .../prometheus/templates/clusterrole.yaml | 56 + .../templates/clusterrolebinding.yaml | 16 + .../charts/prometheus/templates/cm.yaml | 103 + .../charts/prometheus/templates/deploy.yaml | 410 ++ .../prometheus/templates/extra-manifests.yaml | 4 + .../prometheus/templates/headless-svc.yaml | 35 + .../charts/prometheus/templates/ingress.yaml | 57 + .../prometheus/templates/network-policy.yaml | 16 + .../charts/prometheus/templates/pdb.yaml | 15 + .../charts/prometheus/templates/psp.yaml | 53 + .../charts/prometheus/templates/pvc.yaml | 43 + .../prometheus/templates/rolebinding.yaml | 18 + .../charts/prometheus/templates/service.yaml | 63 + .../prometheus/templates/serviceaccount.yaml | 16 + .../charts/prometheus/templates/sts.yaml | 436 ++ .../charts/prometheus/templates/vpa.yaml | 26 + .../charts/prometheus/values.schema.json | 752 ++ .../7.0.1401/charts/prometheus/values.yaml | 1315 ++++ charts/kasten/k10/7.0.1401/config.json | 0 charts/kasten/k10/7.0.1401/eula.txt | 459 ++ charts/kasten/k10/7.0.1401/files/favicon.png | Bin 0 -> 1802 bytes .../kasten/k10/7.0.1401/files/kasten-logo.svg | 24 + charts/kasten/k10/7.0.1401/files/styles.css | 113 + .../grafana/dashboards/default/default.json | 6337 +++++++++++++++++ charts/kasten/k10/7.0.1401/license | 1 + charts/kasten/k10/7.0.1401/questions.yaml | 295 + .../kasten/k10/7.0.1401/templates/NOTES.txt | 222 + .../k10/7.0.1401/templates/_definitions.tpl | 256 + .../k10/7.0.1401/templates/_grafana.tpl | 18 + .../k10/7.0.1401/templates/_helpers.tpl | 1614 +++++ .../k10/7.0.1401/templates/_k10_container.tpl | 1125 +++ .../k10/7.0.1401/templates/_k10_image_tag.tpl | 1 + .../k10/7.0.1401/templates/_k10_metering.tpl | 352 + .../7.0.1401/templates/_k10_serviceimage.tpl | 50 + .../k10/7.0.1401/templates/_k10_template.tpl | 245 + .../k10/7.0.1401/templates/_prometheus.tpl | 29 + .../templates/aggregatedaudit-policy.yaml | 34 + .../7.0.1401/templates/api-tls-secrets.yaml | 13 + .../k10/7.0.1401/templates/apiservice.yaml | 25 + .../k10/7.0.1401/templates/daemonsets.yaml | 26 + .../k10/7.0.1401/templates/deployments.yaml | 39 + .../templates/fluentbit-configmap.yaml | 34 + .../templates/frontend-nginx-configmap.yaml | 50 + .../k10/7.0.1401/templates/gateway-ext.yaml | 36 + .../k10/7.0.1401/templates/gateway.yaml | 249 + .../k10/7.0.1401/templates/grafana-scc.yaml | 45 + .../k10/7.0.1401/templates/ingress.yaml | 73 + .../k10/7.0.1401/templates/k10-config.yaml | 355 + .../k10/7.0.1401/templates/k10-eula.yaml | 21 + .../k10/7.0.1401/templates/k10-scc.yaml | 46 + .../7.0.1401/templates/kopia-tls-certs.yaml | 33 + .../k10/7.0.1401/templates/license.yaml | 25 + charts/kasten/k10/7.0.1401/templates/mc.yaml | 6 + .../7.0.1401/templates/mutatingwebhook.yaml | 51 + .../k10/7.0.1401/templates/networkpolicy.yaml | 299 + .../templates/ocp-ca-cert-extract-hook.yaml | 218 + .../7.0.1401/templates/ocp-consoleplugin.yaml | 27 + .../templates/ocp-plugin-configmap.yaml | 28 + .../templates/ocp-plugin-deployment.yaml | 74 + .../templates/ocp-plugin-proxy-configmap.yaml | 36 + .../ocp-plugin-proxy-deployment.yaml | 67 + .../templates/ocp-plugin-proxy-service.yaml | 25 + .../templates/ocp-plugin-service.yaml | 23 + .../templates/prometheus-configmap.yaml | 97 + .../7.0.1401/templates/prometheus-scc.yaml | 41 + .../templates/prometheus-service.yaml | 46 + .../kasten/k10/7.0.1401/templates/rbac.yaml | 381 + .../kasten/k10/7.0.1401/templates/route.yaml | 36 + .../k10/7.0.1401/templates/secrets.yaml | 257 + .../7.0.1401/templates/secure_deployment.tpl | 17 + .../7.0.1401/templates/serviceaccount.yaml | 48 + .../k10/7.0.1401/templates/v0services.yaml | 200 + .../templates/workloadIdentityFederation.tpl | 10 + .../grafana/values/grafana_values.tpl | 278 + .../{charts}/values/prometheus_values.tpl | 186 + charts/kasten/k10/7.0.1401/triallicense | 1 + charts/kasten/k10/7.0.1401/values.schema.json | 3126 ++++++++ charts/kasten/k10/7.0.1401/values.yaml | 606 ++ charts/kuma/kuma/2.9.1/.helmdocsignore | 1 + charts/kuma/kuma/2.9.1/.helmignore | 23 + charts/kuma/kuma/2.9.1/Chart.yaml | 26 + charts/kuma/kuma/2.9.1/README.md | 316 + charts/kuma/kuma/2.9.1/README.md.gotmpl | 52 + .../2.9.1/crds/kuma.io_circuitbreakers.yaml | 50 + .../2.9.1/crds/kuma.io_containerpatches.yaml | 114 + .../2.9.1/crds/kuma.io_dataplaneinsights.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_dataplanes.yaml | 70 + .../2.9.1/crds/kuma.io_externalservices.yaml | 50 + .../2.9.1/crds/kuma.io_faultinjections.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_healthchecks.yaml | 50 + .../crds/kuma.io_hostnamegenerators.yaml | 72 + .../2.9.1/crds/kuma.io_meshaccesslogs.yaml | 557 ++ .../crds/kuma.io_meshcircuitbreakers.yaml | 739 ++ .../kuma/kuma/2.9.1/crds/kuma.io_meshes.yaml | 50 + .../crds/kuma.io_meshexternalservices.yaml | 333 + .../crds/kuma.io_meshfaultinjections.yaml | 420 ++ .../crds/kuma.io_meshgatewayinstances.yaml | 354 + .../2.9.1/crds/kuma.io_meshgatewayroutes.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_meshgateways.yaml | 50 + .../2.9.1/crds/kuma.io_meshhealthchecks.yaml | 382 + .../2.9.1/crds/kuma.io_meshhttproutes.yaml | 668 ++ .../kuma/2.9.1/crds/kuma.io_meshinsights.yaml | 50 + .../kuma.io_meshloadbalancingstrategies.yaml | 572 ++ .../kuma/2.9.1/crds/kuma.io_meshmetrics.yaml | 292 + .../crds/kuma.io_meshmultizoneservices.yaml | 199 + .../2.9.1/crds/kuma.io_meshpassthroughs.yaml | 164 + .../2.9.1/crds/kuma.io_meshproxypatches.yaml | 550 ++ .../2.9.1/crds/kuma.io_meshratelimits.yaml | 499 ++ .../kuma/2.9.1/crds/kuma.io_meshretries.yaml | 507 ++ .../kuma/2.9.1/crds/kuma.io_meshservices.yaml | 218 + .../2.9.1/crds/kuma.io_meshtcproutes.yaml | 282 + .../kuma/2.9.1/crds/kuma.io_meshtimeouts.yaml | 363 + .../kuma/2.9.1/crds/kuma.io_meshtlses.yaml | 239 + .../kuma/2.9.1/crds/kuma.io_meshtraces.yaml | 283 + .../crds/kuma.io_meshtrafficpermissions.yaml | 203 + .../2.9.1/crds/kuma.io_proxytemplates.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_ratelimits.yaml | 50 + .../kuma/kuma/2.9.1/crds/kuma.io_retries.yaml | 50 + .../2.9.1/crds/kuma.io_serviceinsights.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_timeouts.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_trafficlogs.yaml | 50 + .../crds/kuma.io_trafficpermissions.yaml | 50 + .../2.9.1/crds/kuma.io_trafficroutes.yaml | 50 + .../2.9.1/crds/kuma.io_traffictraces.yaml | 50 + .../2.9.1/crds/kuma.io_virtualoutbounds.yaml | 50 + .../kuma/2.9.1/crds/kuma.io_zoneegresses.yaml | 56 + .../crds/kuma.io_zoneegressinsights.yaml | 50 + .../2.9.1/crds/kuma.io_zoneingresses.yaml | 56 + .../crds/kuma.io_zoneingressinsights.yaml | 51 + .../kuma/2.9.1/crds/kuma.io_zoneinsights.yaml | 50 + .../kuma/kuma/2.9.1/crds/kuma.io_zones.yaml | 50 + charts/kuma/kuma/2.9.1/templates/NOTES.txt | 42 + charts/kuma/kuma/2.9.1/templates/_helpers.tpl | 432 ++ .../kuma/2.9.1/templates/cni-configmap.yaml | 22 + .../kuma/2.9.1/templates/cni-daemonset.yaml | 152 + .../kuma/kuma/2.9.1/templates/cni-rbac.yaml | 51 + .../kuma/2.9.1/templates/cp-configmap.yaml | 46 + .../kuma/2.9.1/templates/cp-deployment.yaml | 412 ++ .../templates/cp-global-sync-service.yaml | 33 + charts/kuma/kuma/2.9.1/templates/cp-hpa.yaml | 24 + .../kuma/kuma/2.9.1/templates/cp-ingress.yaml | 25 + .../cp-kds-global-server-secret.yaml | 15 + .../cp-kds-zone-client-tls-secret.yaml | 13 + charts/kuma/kuma/2.9.1/templates/cp-pdb.yaml | 20 + charts/kuma/kuma/2.9.1/templates/cp-rbac.yaml | 320 + .../kuma/kuma/2.9.1/templates/cp-service.yaml | 49 + .../templates/cp-webhooks-and-secrets.yaml | 346 + .../2.9.1/templates/egress-deployment.yaml | 138 + .../kuma/kuma/2.9.1/templates/egress-hpa.yaml | 24 + .../kuma/kuma/2.9.1/templates/egress-pdb.yaml | 20 + .../kuma/2.9.1/templates/egress-rbac.yaml | 18 + .../kuma/2.9.1/templates/egress-service.yaml | 32 + .../kuma/2.9.1/templates/gateway-class.yaml | 19 + .../2.9.1/templates/ingress-deployment.yaml | 142 + .../kuma/2.9.1/templates/ingress-hpa.yaml | 24 + .../kuma/2.9.1/templates/ingress-pdb.yaml | 20 + .../kuma/2.9.1/templates/ingress-rbac.yaml | 18 + .../kuma/2.9.1/templates/ingress-service.yaml | 32 + .../post-delete-cleanup-ebpf-job.yaml | 126 + .../2.9.1/templates/pre-delete-webhooks.yaml | 109 + .../pre-install-patch-namespace-job.yaml | 124 + .../pre-upgrade-install-crds-job.yaml | 171 + charts/kuma/kuma/2.9.1/values.yaml | 903 +++ .../2.2.10/Chart.yaml | 21 + .../2.2.10/README.md | 747 ++ .../2.2.10/app-readme.md | 5 + .../2.2.10/crds/crds.yaml | 2706 +++++++ .../2.2.10/templates/NOTES.txt | 14 + .../2.2.10/templates/_helpers.tpl | 93 + .../2.2.10/templates/configmap.yaml | 110 + .../2.2.10/templates/deployment.yaml | 479 ++ .../2.2.10/templates/ingressclass.yaml | 18 + .../2.2.10/templates/rbac.yaml | 106 + .../2.2.10/values.yaml | 306 + .../2.2.10/Chart.yaml | 20 + .../2.2.10/README.md | 533 ++ .../2.2.10/app-readme.md | 5 + .../2.2.10/crds/crds.yaml | 2706 +++++++ .../2.2.10/templates/NOTES.txt | 15 + .../2.2.10/templates/_helpers.tpl | 79 + .../2.2.10/templates/configmap.yaml | 82 + .../2.2.10/templates/deployment.yaml | 282 + .../2.2.10/templates/ingressclass.yaml | 18 + .../2.2.10/templates/rbac.yaml | 106 + .../2.2.10/values.yaml | 230 + .../5.0.0/.helmignore | 23 + .../5.0.0/BUILD.bazel | 9 + .../k8s-triliovault-operator/5.0.0/Chart.yaml | 24 + .../k8s-triliovault-operator/5.0.0/LICENSE | 1 + .../k8s-triliovault-operator/5.0.0/README.md | 206 + .../5.0.0/charts/observability/Chart.yaml | 21 + .../observability/charts/logging/Chart.yaml | 18 + .../charts/logging/charts/loki/Chart.yaml | 13 + .../logging/charts/loki/templates/NOTES.txt | 3 + .../charts/loki/templates/_helpers.tpl | 99 + .../loki/templates/configmap-alert.yaml | 14 + .../charts/loki/templates/ingress.yaml | 52 + .../charts/loki/templates/networkpolicy.yaml | 23 + .../logging/charts/loki/templates/pdb.yaml | 14 + .../loki/templates/podsecuritypolicy.yaml | 40 + .../charts/loki/templates/prometheusrule.yaml | 20 + .../logging/charts/loki/templates/role.yaml | 17 + .../charts/loki/templates/rolebinding.yaml | 17 + .../logging/charts/loki/templates/secret.yaml | 11 + .../loki/templates/service-headless.yaml | 26 + .../loki/templates/service-memberlist.yaml | 21 + .../charts/loki/templates/service.yaml | 43 + .../charts/loki/templates/serviceaccount.yaml | 13 + .../charts/loki/templates/servicemonitor.yaml | 42 + .../charts/loki/templates/statefulset.yaml | 180 + .../charts/logging/charts/loki/values.yaml | 346 + .../charts/logging/charts/promtail/Chart.yaml | 16 + .../charts/promtail/templates/NOTES.txt | 15 + .../charts/promtail/templates/_helpers.tpl | 102 + .../charts/promtail/templates/_pod.tpl | 118 + .../promtail/templates/clusterrole.yaml | 21 + .../templates/clusterrolebinding.yaml | 16 + .../charts/promtail/templates/configmap.yaml | 12 + .../charts/promtail/templates/daemonset.yaml | 21 + .../charts/promtail/templates/deployment.yaml | 22 + .../promtail/templates/extra-manifests.yaml | 4 + .../charts/promtail/templates/hpa.yaml | 31 + .../promtail/templates/networkpolicy.yaml | 123 + .../promtail/templates/podsecuritypolicy.yaml | 10 + .../promtail/templates/prometheus-rules.yaml | 21 + .../charts/promtail/templates/role.yaml | 18 + .../promtail/templates/rolebinding.yaml | 16 + .../charts/promtail/templates/secret.yaml | 19 + .../promtail/templates/service-extra.yaml | 52 + .../promtail/templates/service-metrics.yaml | 18 + .../promtail/templates/serviceaccount.yaml | 17 + .../promtail/templates/servicemonitor.yaml | 58 + .../logging/charts/promtail/values.yaml | 534 ++ .../charts/logging/templates/_helpers.tpl | 50 + .../charts/logging/templates/datasources.yaml | 24 + .../charts/monitoring/Chart.yaml | 16 + .../monitoring/charts/prometheus/Chart.yaml | 22 + .../charts/kube-state-metrics/Chart.yaml | 17 + .../kube-state-metrics/templates/NOTES.txt | 10 + .../kube-state-metrics/templates/_helpers.tpl | 82 + .../templates/clusterrolebinding.yaml | 20 + .../templates/deployment.yaml | 156 + .../templates/kubeconfig-secret.yaml | 12 + .../kube-state-metrics/templates/pdb.yaml | 14 + .../templates/podsecuritypolicy.yaml | 39 + .../templates/psp-clusterrole.yaml | 19 + .../templates/psp-clusterrolebinding.yaml | 16 + .../kube-state-metrics/templates/role.yaml | 187 + .../templates/rolebinding.yaml | 24 + .../kube-state-metrics/templates/service.yaml | 38 + .../templates/serviceaccount.yaml | 15 + .../templates/servicemonitor.yaml | 66 + .../templates/stsdiscovery-role.yaml | 26 + .../templates/stsdiscovery-rolebinding.yaml | 17 + .../charts/kube-state-metrics/values.yaml | 233 + .../charts/prometheus/templates/NOTES.txt | 112 + .../charts/prometheus/templates/_helpers.tpl | 288 + .../templates/alertmanager/clusterrole.yaml | 21 + .../alertmanager/clusterrolebinding.yaml | 20 + .../prometheus/templates/alertmanager/cm.yaml | 19 + .../templates/alertmanager/deploy.yaml | 208 + .../templates/alertmanager/headless-svc.yaml | 31 + .../templates/alertmanager/ingress.yaml | 57 + .../templates/alertmanager/netpol.yaml | 20 + .../templates/alertmanager/pdb.yaml | 14 + .../templates/alertmanager/psp.yaml | 46 + .../templates/alertmanager/pvc.yaml | 37 + .../templates/alertmanager/role.yaml | 24 + .../templates/alertmanager/rolebinding.yaml | 23 + .../templates/alertmanager/service.yaml | 53 + .../alertmanager/serviceaccount.yaml | 11 + .../templates/alertmanager/sts.yaml | 188 + .../templates/node-exporter/daemonset.yaml | 150 + .../templates/node-exporter/psp.yaml | 55 + .../templates/node-exporter/role.yaml | 17 + .../templates/node-exporter/rolebinding.yaml | 19 + .../node-exporter/serviceaccount.yaml | 11 + .../templates/node-exporter/svc.yaml | 47 + .../templates/pushgateway/clusterrole.yaml | 21 + .../pushgateway/clusterrolebinding.yaml | 16 + .../templates/pushgateway/deploy.yaml | 119 + .../templates/pushgateway/ingress.yaml | 54 + .../templates/pushgateway/netpol.yaml | 20 + .../prometheus/templates/pushgateway/pdb.yaml | 14 + .../prometheus/templates/pushgateway/psp.yaml | 42 + .../prometheus/templates/pushgateway/pvc.yaml | 31 + .../templates/pushgateway/service.yaml | 41 + .../templates/pushgateway/serviceaccount.yaml | 11 + .../prometheus/templates/pushgateway/vpa.yaml | 20 + .../templates/server/clusterrole.yaml | 48 + .../templates/server/clusterrolebinding.yaml | 16 + .../prometheus/templates/server/cm.yaml | 85 + .../prometheus/templates/server/deploy.yaml | 324 + .../templates/server/extra-manifests.yaml | 4 + .../templates/server/headless-svc.yaml | 37 + .../prometheus/templates/server/ingress.yaml | 59 + .../prometheus/templates/server/netpol.yaml | 18 + .../prometheus/templates/server/pdb.yaml | 14 + .../prometheus/templates/server/psp.yaml | 51 + .../prometheus/templates/server/pvc.yaml | 39 + .../templates/server/rolebinding.yaml | 20 + .../prometheus/templates/server/service.yaml | 60 + .../templates/server/serviceaccount.yaml | 13 + .../prometheus/templates/server/sts.yaml | 302 + .../prometheus/templates/server/vpa.yaml | 24 + .../monitoring/charts/prometheus/values.yaml | 1861 +++++ .../charts/monitoring/templates/_helpers.tpl | 49 + .../monitoring/templates/datasources.yaml | 25 + .../charts/visualization/Chart.yaml | 16 + .../visualization/charts/grafana/Chart.yaml | 14 + .../grafana/dashboards/backup-detail.json | 926 +++ .../grafana/dashboards/backup-overview.json | 883 +++ .../grafana/dashboards/backupplan-detail.json | 1198 ++++ .../dashboards/backupplan-overview.json | 883 +++ .../dashboards/clusterbackup-detail.json | 820 +++ .../dashboards/clusterbackupplan-detail.json | 1234 ++++ .../dashboards/clusterrestore-detail.json | 802 +++ .../dashboards/consistentset-detail.json | 832 +++ .../continuousrestoreplan-detail.json | 1052 +++ .../grafana/dashboards/logging-dashboard.json | 212 + .../grafana/dashboards/metadata-detail.json | 889 +++ .../charts/grafana/dashboards/overview.json | 1429 ++++ .../grafana/dashboards/restore-detail.json | 901 +++ .../grafana/dashboards/restore-overview.json | 853 +++ .../grafana/dashboards/target-detail.json | 1327 ++++ .../charts/grafana/templates/NOTES.txt | 54 + .../charts/grafana/templates/_helpers.tpl | 165 + .../charts/grafana/templates/_pod.tpl | 748 ++ .../charts/grafana/templates/clusterrole.yaml | 25 + .../grafana/templates/clusterrolebinding.yaml | 24 + .../configmap-dashboard-provider.yaml | 29 + .../charts/grafana/templates/configmap.yaml | 88 + .../templates/dashboards-json-configmap.yaml | 36 + .../charts/grafana/templates/deployment.yaml | 50 + .../grafana/templates/extra-manifests.yaml | 4 + .../grafana/templates/headless-service.yaml | 22 + .../charts/grafana/templates/hpa.yaml | 20 + .../templates/image-renderer-deployment.yaml | 121 + .../image-renderer-network-policy.yaml | 76 + .../templates/image-renderer-service.yaml | 30 + .../charts/grafana/templates/ingress.yaml | 78 + .../grafana/templates/networkpolicy.yaml | 37 + .../templates/poddisruptionbudget.yaml | 22 + .../grafana/templates/podsecuritypolicy.yaml | 49 + .../charts/grafana/templates/pvc.yaml | 33 + .../charts/grafana/templates/role.yaml | 32 + .../charts/grafana/templates/rolebinding.yaml | 25 + .../charts/grafana/templates/secret-env.yaml | 14 + .../charts/grafana/templates/secret.yaml | 26 + .../charts/grafana/templates/service.yaml | 51 + .../grafana/templates/serviceaccount.yaml | 14 + .../grafana/templates/servicemonitor.yaml | 44 + .../charts/grafana/templates/statefulset.yaml | 52 + .../charts/grafana/templates/user-secret.yaml | 9 + .../visualization/charts/grafana/values.yaml | 938 +++ ...iovault.trilio.io_triliovaultmanagers.yaml | 1231 ++++ .../5.0.0/questions.yaml | 158 + .../5.0.0/templates/NOTES.txt | 59 + .../5.0.0/templates/TVMCustomResource.yaml | 65 + .../5.0.0/templates/TVMSecret.yaml | 25 + .../5.0.0/templates/_helpers.tpl | 134 + .../5.0.0/templates/clusterrole.yaml | 148 + .../5.0.0/templates/clusterrole_binding.yaml | 15 + .../5.0.0/templates/deployment.yaml | 390 + .../5.0.0/templates/mutating-webhook.yaml | 28 + .../preflight_job_preinstall_hook.yaml | 203 + .../5.0.0/templates/secret.yaml | 9 + .../5.0.0/templates/serviceAccount.yaml | 14 + .../5.0.0/templates/validating-webhook.yaml | 104 + .../5.0.0/templates/webhook-service.yaml | 17 + .../5.0.0/values.yaml | 247 + index.yaml | 233 +- 881 files changed, 97624 insertions(+), 44362 deletions(-) create mode 100644 assets/airlock/microgateway-4.4.1.tgz create mode 100644 assets/airlock/microgateway-cni-4.4.1.tgz create mode 100644 assets/kasten/k10-7.0.1401.tgz create mode 100644 assets/kuma/kuma-2.9.1.tgz create mode 100644 assets/netscaler/netscaler-cpx-with-ingress-controller-2.2.10.tgz create mode 100644 assets/netscaler/netscaler-ingress-controller-2.2.10.tgz create mode 100644 assets/trilio/k8s-triliovault-operator-5.0.0.tgz create mode 100644 charts/airlock/microgateway-cni/4.4.1/.helmignore create mode 100644 charts/airlock/microgateway-cni/4.4.1/Chart.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/README.md create mode 100644 charts/airlock/microgateway-cni/4.4.1/gke-values.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/openshift-values.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/questions.yml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/NOTES.txt create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/_helpers.tpl create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/configmap.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/network-attachment-definition.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/serviceaccount.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/tests/rbac.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/templates/tests/test-install.yaml create mode 100644 charts/airlock/microgateway-cni/4.4.1/values.schema.json create mode 100644 charts/airlock/microgateway-cni/4.4.1/values.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/accesscontrols.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/contentsecurities.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/denyrules.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/envoyclusters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/envoyconfigurations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/headerrewrites.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/identitypropagations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/limits.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/oidcproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/openapis.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/parsers.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/sidecargateways.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/crds/telemetries.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.2.3/gke-values.yaml create mode 100644 charts/airlock/microgateway/4.2.3/openshift-values.yaml create mode 100644 charts/airlock/microgateway/4.2.3/questions.yml create mode 100644 charts/airlock/microgateway/4.2.3/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/configmap.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/network-attachment-definition.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/_webhooks.gen.tpl delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/configmap.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/deployment.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrole.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/mutating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/podmonitor.yaml delete mode 100644 charts/airlock/microgateway/4.2.3/templates/operator/validating-webhook.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.2.3/templates/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/app-readme.md delete mode 100644 charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/dashboards/license.json delete mode 100644 charts/airlock/microgateway/4.3.0/dashboards/overview.json create mode 100644 charts/airlock/microgateway/4.3.0/gke-values.yaml create mode 100644 charts/airlock/microgateway/4.3.0/openshift-values.yaml create mode 100644 charts/airlock/microgateway/4.3.0/questions.yml create mode 100644 charts/airlock/microgateway/4.3.0/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/network-attachment-definition.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/role.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/service.yaml delete mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/app-readme.md delete mode 100644 charts/airlock/microgateway/4.3.1/crds/accesscontrols.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/contentsecurities.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/denyrules.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/graphqls.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/headerrewrites.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/identitypropagations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/limits.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/oidcproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/openapis.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/parsers.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/redisproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/sessionhandlings.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/crds/telemetries.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/dashboards/license.json delete mode 100644 charts/airlock/microgateway/4.3.1/dashboards/overview.json create mode 100644 charts/airlock/microgateway/4.3.1/gke-values.yaml create mode 100644 charts/airlock/microgateway/4.3.1/openshift-values.yaml create mode 100644 charts/airlock/microgateway/4.3.1/questions.yml create mode 100644 charts/airlock/microgateway/4.3.1/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/network-attachment-definition.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/_operator_helpers.tpl delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/_rbac.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/dashboard-configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/deployment.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/manager-role.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/manager-rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/metrics-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/mutating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/podmonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/role.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/selfsigned-issuer.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/servicemonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/serving-certificate.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/validating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/webhook-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/operator/xds-service.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.1/templates/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/tests/service.yaml delete mode 100644 charts/airlock/microgateway/4.3.1/templates/tests/statefulset.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/app-readme.md delete mode 100644 charts/airlock/microgateway/4.3.3/crds/accesscontrols.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/contentsecurities.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/denyrules.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/envoyclusters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/envoyconfigurations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/headerrewrites.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/identitypropagations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/limits.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/oidcproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/openapis.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/redisproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/sessionhandlings.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/sidecargateways.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/crds/telemetries.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/dashboards/blockMetrics.json delete mode 100644 charts/airlock/microgateway/4.3.3/dashboards/license.json create mode 100644 charts/airlock/microgateway/4.3.3/gke-values.yaml create mode 100644 charts/airlock/microgateway/4.3.3/openshift-values.yaml create mode 100644 charts/airlock/microgateway/4.3.3/questions.yml create mode 100644 charts/airlock/microgateway/4.3.3/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/network-attachment-definition.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/_operator_helpers.tpl delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/_rbac.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/_webhooks.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/dashboard-configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/deployment.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/manager-role.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/manager-rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/metrics-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/mutating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/podmonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/role.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/selfsigned-issuer.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/servicemonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/serving-certificate.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/validating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/webhook-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/operator/xds-service.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.3/templates/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/tests/service.yaml delete mode 100644 charts/airlock/microgateway/4.3.3/templates/tests/statefulset.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/app-readme.md delete mode 100644 charts/airlock/microgateway/4.3.4/crds/accesscontrols.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/contentsecurities.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/denyrules.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/envoyclusters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/envoyconfigurations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/envoyhttpfilters.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/graphqls.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/headerrewrites.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/identitypropagations.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/limits.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/oidcproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/oidcrelyingparties.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/openapis.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/parsers.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/redisproviders.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/sessionhandlings.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/sidecargateways.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/crds/telemetries.microgateway.airlock.com.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/dashboards/blockLogs.json delete mode 100644 charts/airlock/microgateway/4.3.4/dashboards/blockMetrics.json delete mode 100644 charts/airlock/microgateway/4.3.4/dashboards/license.json create mode 100644 charts/airlock/microgateway/4.3.4/gke-values.yaml create mode 100644 charts/airlock/microgateway/4.3.4/openshift-values.yaml create mode 100644 charts/airlock/microgateway/4.3.4/questions.yml create mode 100644 charts/airlock/microgateway/4.3.4/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/network-attachment-definition.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/_operator_helpers.tpl delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/_rbac.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/_webhooks.gen.tpl delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/dashboard-configmap.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/deployment.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/manager-role.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/manager-rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/metrics-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/mutating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/podmonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/role.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/rolebinding.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/selfsigned-issuer.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/servicemonitor.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/serving-certificate.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/validating-webhook.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/webhook-service.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/operator/xds-service.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.4/templates/serviceaccount.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/tests/service.yaml delete mode 100644 charts/airlock/microgateway/4.3.4/templates/tests/statefulset.yaml create mode 100644 charts/airlock/microgateway/4.4.1/.helmignore create mode 100644 charts/airlock/microgateway/4.4.1/Chart.yaml create mode 100644 charts/airlock/microgateway/4.4.1/README.md rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/app-readme.md (100%) create mode 100644 charts/airlock/microgateway/4.4.1/crds/accesscontrols.microgateway.airlock.com.yaml rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/contentsecurities.microgateway.airlock.com.yaml (98%) create mode 100644 charts/airlock/microgateway/4.4.1/crds/contentsecuritypolicies.microgateway.airlock.com.yaml rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/denyrules.microgateway.airlock.com.yaml (99%) rename charts/airlock/microgateway/{4.3.1 => 4.4.1}/crds/envoyclusters.microgateway.airlock.com.yaml (96%) rename charts/airlock/microgateway/{4.3.1 => 4.4.1}/crds/envoyconfigurations.microgateway.airlock.com.yaml (98%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/crds/envoyhttpfilters.microgateway.airlock.com.yaml (96%) rename charts/airlock/microgateway/{4.3.3 => 4.4.1}/crds/graphqls.microgateway.airlock.com.yaml (97%) create mode 100644 charts/airlock/microgateway/4.4.1/crds/headerrewrites.microgateway.airlock.com.yaml rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/identitypropagations.microgateway.airlock.com.yaml (64%) create mode 100644 charts/airlock/microgateway/4.4.1/crds/jwks.microgateway.airlock.com.yaml rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/limits.microgateway.airlock.com.yaml (99%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/oidcproviders.microgateway.airlock.com.yaml (86%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/oidcrelyingparties.microgateway.airlock.com.yaml (84%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/openapis.microgateway.airlock.com.yaml (98%) rename charts/airlock/microgateway/{4.3.3 => 4.4.1}/crds/parsers.microgateway.airlock.com.yaml (99%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/redisproviders.microgateway.airlock.com.yaml (67%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/sessionhandlings.microgateway.airlock.com.yaml (57%) rename charts/airlock/microgateway/{4.3.1 => 4.4.1}/crds/sidecargateways.microgateway.airlock.com.yaml (99%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/crds/telemetries.microgateway.airlock.com.yaml (98%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/dashboards/blockLogs.json (68%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/dashboards/blockMetrics.json (95%) rename charts/airlock/microgateway/{4.3.3/dashboards/blockLogs.json => 4.4.1/dashboards/headerLogs.json} (55%) rename charts/airlock/microgateway/{4.3.4/dashboards/overview.json => 4.4.1/dashboards/license.json} (54%) rename charts/airlock/microgateway/{4.3.1/dashboards/blockLogs.json => 4.4.1/dashboards/logOnlyLogs.json} (55%) rename charts/airlock/microgateway/{4.3.1/dashboards/blockMetrics.json => 4.4.1/dashboards/logOnlyMetrics.json} (70%) rename charts/airlock/microgateway/{4.3.3 => 4.4.1}/dashboards/overview.json (98%) create mode 100644 charts/airlock/microgateway/4.4.1/templates/NOTES.txt create mode 100644 charts/airlock/microgateway/4.4.1/templates/_helpers.tpl rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/_operator_helpers.tpl (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/_rbac.gen.tpl (73%) rename charts/airlock/microgateway/{4.3.1 => 4.4.1}/templates/operator/_webhooks.gen.tpl (84%) rename charts/airlock/microgateway/{4.3.1 => 4.4.1}/templates/operator/configmap.yaml (97%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/dashboard-configmap.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/deployment.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/manager-role.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/manager-rolebinding.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/metrics-service.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/mutating-webhook.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/podmonitor.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/role.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/rolebinding.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/selfsigned-issuer.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/serviceaccount.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/servicemonitor.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/serving-certificate.yaml (100%) rename charts/airlock/microgateway/{4.3.0 => 4.4.1}/templates/operator/validating-webhook.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/webhook-service.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/operator/xds-service.yaml (100%) create mode 100644 charts/airlock/microgateway/4.4.1/templates/tests/rbac.yaml rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/tests/service.yaml (100%) rename charts/airlock/microgateway/{4.2.3 => 4.4.1}/templates/tests/statefulset.yaml (100%) create mode 100644 charts/airlock/microgateway/4.4.1/templates/tests/test-install.yaml create mode 100644 charts/airlock/microgateway/4.4.1/values.schema.json create mode 100644 charts/airlock/microgateway/4.4.1/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/Chart.lock create mode 100644 charts/kasten/k10/7.0.1401/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/README.md create mode 100644 charts/kasten/k10/7.0.1401/app-readme.md create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/default-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-affinity-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-json-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-extraconfigmapmounts-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-image-renderer-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-nondefault-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-persistence.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/dashboards/custom-dashboard.json create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/_config.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/_pod.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/configSecret.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap-dashboard-provider.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/dashboards-json-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/headless-service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/hpa.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-hpa.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-network-policy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/poddisruptionbudget.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/pvc.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/role.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/secret-env.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/secret.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-role.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/grafana/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/Chart.lock create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/OWNERS create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/services.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/vpa.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.schema.json create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/role.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/common-labels-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/default-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/networkpolicy-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/pod-labels-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/service-labels-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/serviceport-values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/.helmignore create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/README.md create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/cm.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/deploy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/headless-svc.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/network-policy.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/psp.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/pvc.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/sts.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/templates/vpa.yaml create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/values.schema.json create mode 100644 charts/kasten/k10/7.0.1401/charts/prometheus/values.yaml create mode 100644 charts/kasten/k10/7.0.1401/config.json create mode 100644 charts/kasten/k10/7.0.1401/eula.txt create mode 100644 charts/kasten/k10/7.0.1401/files/favicon.png create mode 100644 charts/kasten/k10/7.0.1401/files/kasten-logo.svg create mode 100644 charts/kasten/k10/7.0.1401/files/styles.css create mode 100644 charts/kasten/k10/7.0.1401/grafana/dashboards/default/default.json create mode 100644 charts/kasten/k10/7.0.1401/license create mode 100644 charts/kasten/k10/7.0.1401/questions.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1401/templates/_definitions.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_grafana.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_k10_container.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_k10_image_tag.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_k10_metering.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_k10_serviceimage.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_k10_template.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/_prometheus.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/aggregatedaudit-policy.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/api-tls-secrets.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/apiservice.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/daemonsets.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/deployments.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/fluentbit-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/frontend-nginx-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/gateway-ext.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/gateway.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/grafana-scc.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/k10-config.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/k10-eula.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/k10-scc.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/kopia-tls-certs.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/license.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/mc.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/mutatingwebhook.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-ca-cert-extract-hook.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-consoleplugin.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-deployment.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-service.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/ocp-plugin-service.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/prometheus-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/prometheus-scc.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/prometheus-service.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/rbac.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/route.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/secrets.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/secure_deployment.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/v0services.yaml create mode 100644 charts/kasten/k10/7.0.1401/templates/workloadIdentityFederation.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/{values}/grafana/values/grafana_values.tpl create mode 100644 charts/kasten/k10/7.0.1401/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl create mode 100644 charts/kasten/k10/7.0.1401/triallicense create mode 100644 charts/kasten/k10/7.0.1401/values.schema.json create mode 100644 charts/kasten/k10/7.0.1401/values.yaml create mode 100644 charts/kuma/kuma/2.9.1/.helmdocsignore create mode 100644 charts/kuma/kuma/2.9.1/.helmignore create mode 100644 charts/kuma/kuma/2.9.1/Chart.yaml create mode 100644 charts/kuma/kuma/2.9.1/README.md create mode 100644 charts/kuma/kuma/2.9.1/README.md.gotmpl create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_circuitbreakers.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_containerpatches.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_dataplaneinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_dataplanes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_externalservices.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_faultinjections.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_healthchecks.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_hostnamegenerators.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshaccesslogs.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshcircuitbreakers.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshexternalservices.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshfaultinjections.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayinstances.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayroutes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshgateways.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshhealthchecks.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshhttproutes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshloadbalancingstrategies.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshmetrics.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshmultizoneservices.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshpassthroughs.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshproxypatches.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshratelimits.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshretries.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshservices.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshtcproutes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshtimeouts.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshtlses.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshtraces.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_meshtrafficpermissions.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_proxytemplates.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_ratelimits.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_retries.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_serviceinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_timeouts.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_trafficlogs.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_trafficpermissions.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_trafficroutes.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_traffictraces.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_virtualoutbounds.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegresses.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegressinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingresses.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingressinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zoneinsights.yaml create mode 100644 charts/kuma/kuma/2.9.1/crds/kuma.io_zones.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/NOTES.txt create mode 100644 charts/kuma/kuma/2.9.1/templates/_helpers.tpl create mode 100644 charts/kuma/kuma/2.9.1/templates/cni-configmap.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cni-daemonset.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cni-rbac.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-configmap.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-deployment.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-global-sync-service.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-hpa.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-ingress.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-kds-global-server-secret.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-kds-zone-client-tls-secret.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-pdb.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-rbac.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-service.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/cp-webhooks-and-secrets.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/egress-deployment.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/egress-hpa.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/egress-pdb.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/egress-rbac.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/egress-service.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/gateway-class.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/ingress-deployment.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/ingress-hpa.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/ingress-pdb.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/ingress-rbac.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/ingress-service.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/post-delete-cleanup-ebpf-job.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/pre-delete-webhooks.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/pre-install-patch-namespace-job.yaml create mode 100644 charts/kuma/kuma/2.9.1/templates/pre-upgrade-install-crds-job.yaml create mode 100644 charts/kuma/kuma/2.9.1/values.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/Chart.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/README.md create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/app-readme.md create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/crds/crds.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/NOTES.txt create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/_helpers.tpl create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/configmap.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/deployment.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/ingressclass.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/rbac.yaml create mode 100644 charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/values.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/Chart.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/README.md create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/app-readme.md create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/crds/crds.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/NOTES.txt create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/_helpers.tpl create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/configmap.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/deployment.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/ingressclass.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/templates/rbac.yaml create mode 100644 charts/netscaler/netscaler-ingress-controller/2.2.10/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/.helmignore create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/BUILD.bazel create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/LICENSE create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/README.md create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/configmap-alert.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/ingress.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/networkpolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/pdb.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/podsecuritypolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/prometheusrule.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-headless.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-memberlist.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/servicemonitor.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/statefulset.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_pod.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/configmap.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/daemonset.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/deployment.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/extra-manifests.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/hpa.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/networkpolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/podsecuritypolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/prometheus-rules.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-extra.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-metrics.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/servicemonitor.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/datasources.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/cm.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/deploy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/headless-svc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/ingress.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/netpol.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pdb.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/psp.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pvc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/sts.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/daemonset.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/psp.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/svc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/deploy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/ingress.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/netpol.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pdb.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/psp.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pvc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/vpa.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/cm.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/deploy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/extra-manifests.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/headless-svc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/ingress.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/netpol.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pdb.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/psp.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pvc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/sts.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/vpa.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/datasources.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/Chart.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-overview.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-overview.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackup-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackupplan-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterrestore-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/consistentset-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/continuousrestoreplan-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/logging-dashboard.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/metadata-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/overview.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-overview.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/target-detail.json create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_pod.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap-dashboard-provider.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/dashboards-json-configmap.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/deployment.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/extra-manifests.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/headless-service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/hpa.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-deployment.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-network-policy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/ingress.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/networkpolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/poddisruptionbudget.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/podsecuritypolicy.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/pvc.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/role.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/rolebinding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret-env.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/serviceaccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/servicemonitor.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/statefulset.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/user-secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/values.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/crds/triliovault.trilio.io_triliovaultmanagers.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/questions.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/NOTES.txt create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMCustomResource.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMSecret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/_helpers.tpl create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole_binding.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/deployment.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/mutating-webhook.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/preflight_job_preinstall_hook.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/secret.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/serviceAccount.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/validating-webhook.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/templates/webhook-service.yaml create mode 100644 charts/trilio/k8s-triliovault-operator/5.0.0/values.yaml diff --git a/assets/airlock/microgateway-4.4.1.tgz b/assets/airlock/microgateway-4.4.1.tgz new file mode 100644 index 0000000000000000000000000000000000000000..e683606491d38454cf467141d36ca509a5051ea2 GIT binary patch literal 82234 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 eV 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 literal 0 HcmV?d00001 diff --git a/assets/airlock/microgateway-cni-4.4.1.tgz b/assets/airlock/microgateway-cni-4.4.1.tgz new file mode 100644 index 0000000000000000000000000000000000000000..049ec58bb5eb53e7b96b754089380771a0d42e01 GIT binary patch literal 10942 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%0A& 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 Q7XSeN|IvePtpJo30L%3?9RL6T literal 0 HcmV?d00001 diff --git a/assets/kuma/kuma-2.9.1.tgz b/assets/kuma/kuma-2.9.1.tgz new file mode 100644 index 0000000000000000000000000000000000000000..511e017f764d505b2d1d9de20438b03bbfb59211 GIT binary patch literal 75599 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
    mFmQ%Z-mN5`&6|HtWl=D_Zo*~j2{4HJ% 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 literal 0 HcmV?d00001 diff --git a/assets/trilio/k8s-triliovault-operator-5.0.0.tgz b/assets/trilio/k8s-triliovault-operator-5.0.0.tgz new file mode 100644 index 0000000000000000000000000000000000000000..772f95f21d727068965969d655bbe7cfa385c7d5 GIT binary patch literal 109779 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%0D5=oX;&O1iV_9yp1ZqTdS~ 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

    Backup Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "max": 100, + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "dark-red", + "value": null + }, + { + "color": "rgb(255, 255, 255)", + "value": 1 + }, + { + "color": "dark-green", + "value": 100 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 7, + "x": 0, + "y": 2 + }, + "id": 45, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": false + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_status_percentage{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "gauge" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 8, + "y": 2 + }, + "id": 50, + "links": [], + "maxDataPoints": 100, + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backup=~\"$Backup\",namespace=~\"$Namespace\",cluster=~\"$Cluster\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "text" + }, + { + "columns": [], + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fontSize": "100%", + "gridPos": { + "h": 14, + "w": 12, + "x": 12, + "y": 2 + }, + "id": 42, + "links": [], + "showHeader": true, + "sort": { + "col": 18, + "desc": true + }, + "styles": [ + { + "$$hashKey": "object:10447", + "alias": "Object Type", + "align": "auto", + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "link": true, + "linkTooltip": "Show Metadata Details", + "linkUrl": "/d/Metadata/metadata-detail?var-Backup=${Backup}&var-ObjectType=${__cell}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}", + "mappingType": 1, + "pattern": "objecttype", + "type": "string" + }, + { + "$$hashKey": "object:1072", + "alias": "Source", + "align": "auto", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "applicationtype", + "thresholds": [], + "type": "string", + "unit": "short" + }, + { + "$$hashKey": "object:1249", + "alias": "Count", + "align": "auto", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "mappingType": 1, + "pattern": "Value", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:10448", + "alias": "", + "align": "right", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "decimals": 2, + "pattern": "/.*/", + "thresholds": [], + "type": "hidden", + "unit": "short" + } + ], + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "avg(trilio_backup_metadata_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}) by (objecttype, applicationtype)", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Metadata Info", + "transform": "table", + "type": "table-old" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 5, + "x": 7, + "y": 3 + }, + "id": 46, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "value" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 6 + }, + "id": 47, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Name", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 6 + }, + "id": 36, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backupplan$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Backup Plan", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "custom": { + "align": "left", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "First" + }, + "properties": [ + { + "id": "displayName", + "value": "Value" + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 49, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Details", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Time", + "applicationtype", + "backup_type", + "completion_ts", + "hook", + "size", + "start_ts", + "target", + "resource_namespace" + ] + } + } + }, + { + "id": "reduce", + "options": { + "reducers": [ + "first" + ] + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "Backup Logs", + "gridPos": { + "h": 11, + "w": 24, + "x": 0, + "y": 16 + }, + "id": 52, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"Backup\",transaction_resource_name=~\"$Backup\",service_type=~\"$service_type\",transaction_resource_namespace=~\"$Namespace\"}", + "refId": "A" + } + ], + "title": "Backup Logs", + "type": "logs" + } + ], + "refresh": "30s", + "schemaVersion": 36, + "style": "dark", + "tags": [ + "logging" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "description": "loki datasource", + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"Backup\"}", + "hide": 0, + "includeAll": false, + "label": "Backup", + "multi": false, + "name": "Backup", + "options": [], + "query": { + "query": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"Backup\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*backup=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=~\"Backup\",transaction_resource_name=~\"$Backup\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=~\"Backup\",transaction_resource_name=~\"$Backup\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=~\"Backup\",transaction_resource_name=~\"$Backup\"},transaction_resource_namespace)", + "description": "Backup Namespace", + "hide": 0, + "includeAll": true, + "label": "Backup Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=~\"Backup\",transaction_resource_name=~\"$Backup\"},transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Backup Detail", + "uid": "Backup", + "version": 1, + "weekStart": "" +} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-overview.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-overview.json new file mode 100644 index 000000000..5d4c209c8 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-overview.json @@ -0,0 +1,883 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:14091", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12600, + "graphTooltip": 0, + "id": null, + "iteration": 1655400242671, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Backups Overview

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(115, 181, 181)", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 2, + "y": 2 + }, + "id": 31, + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^All$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backup_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) ", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "All", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "All", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 6, + "y": 2 + }, + "id": 34, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Available$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backup_info{status=\"Available\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Available", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 10, + "y": 2 + }, + "id": 33, + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Failed$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backup_info{status=\"Failed\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Failed", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "blue", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 14, + "y": 2 + }, + "id": 32, + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^InProgress$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backup_info{status=\"InProgress\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "InProgress", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(129, 135, 135)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 18, + "y": 2 + }, + "id": 37, + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^UnKnown$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backup_info{status=\"UnKnown\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "UnKnown", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "links": [], + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "backup" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Backup Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-Backup=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backupplan" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Plan" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "completion_ts" + }, + "properties": [ + { + "id": "displayName", + "value": "Completion" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "unit", + "value": "time: YYYY-MM-DD HH:mm:ss.SSS" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "target" + }, + "properties": [ + { + "id": "displayName", + "value": "Target" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Target Detail", + "url": "/d/TargetDetail/target-detail?refresh=5s&var-Target=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Percentage" + }, + { + "id": "unit", + "value": "percent" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup_type" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "kind" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Kind" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + } + ] + }, + "gridPos": { + "h": 12, + "w": 24, + "x": 0, + "y": 5 + }, + "id": 29, + "options": { + "footer": { + "enablePagination": false, + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "trilio_backup_status_percentage{status=~\"$Status\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "backup", + "backup_type", + "backupplan", + "completion_ts", + "size", + "status", + "target", + "Value", + "kind" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 1, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backup_info", + "hide": 0, + "includeAll": true, + "label": "Status", + "multi": false, + "name": "Status", + "options": [], + "query": { + "query": "trilio_backup_info", + "refId": "Prometheus-Status-Variable-Query" + }, + "refresh": 1, + "regex": "/.*status=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Backup Overview", + "uid": "BackupOverview", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-detail.json new file mode 100644 index 000000000..2295b2b09 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-detail.json @@ -0,0 +1,1198 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:4254", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12605, + "graphTooltip": 0, + "id": null, + "iteration": 1655401068331, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Backup Plan Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 5, + "x": 0, + "y": 2 + }, + "id": 16, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "Status", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 5, + "y": 2 + }, + "id": 9, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^applicationtype$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Application Type", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "False" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 9, + "y": 2 + }, + "id": 10, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^protected$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Protected", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "0" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 13, + "y": 2 + }, + "id": 13, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup_count$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Num of Backups", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 17, + "y": 2 + }, + "id": 14, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^target$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Target", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "backup" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Backup Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-Backup=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "completion_ts" + }, + "properties": [ + { + "id": "displayName", + "value": "Completion" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup_type" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 5 + }, + "id": 12, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Backup\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "Backups", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "backup_type", + "status", + "backup", + "completion_ts", + "size", + "kind" + ] + } + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "BackupPlan Logs", + "gridPos": { + "h": 14, + "w": 12, + "x": 12, + "y": 5 + }, + "id": 18, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"BackupPlan\",transaction_resource_name=~\"$BackupPlan\",service_type=~\"$service_type\",transaction_resource_namespace=~\"$Namespace\"}", + "refId": "A" + } + ], + "title": "BackupPlan Logs", + "type": "logs" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "consistentset_count" + }, + "properties": [ + { + "id": "displayName", + "value": "ConsistentSet Count" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "ConsistentSet Detail", + "url": "/d/ConsistentSet/?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "continuousrestoreinstance" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestore Instance" + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "continuousrestoreplan" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestorePlan" + }, + { + "id": "custom.align" + }, + { + "id": "links", + "value": [ + { + "title": "ContinuousRestorePlan Detail", + "url": "/d/ContinuousRestorePlan/?refresh=5s&var-ContinuousRestorePlan=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cr_status" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestore Status" + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ContinuousRestore Instance" + }, + "properties": [ + { + "id": "custom.width", + "value": 200 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ConsistentSet" + }, + "properties": [ + { + "id": "custom.width", + "value": 112 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ContinuousRestorePlan" + }, + "properties": [ + { + "id": "custom.width", + "value": 177 + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 12 + }, + "id": 19, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [ + { + "desc": false, + "displayName": "ConsistentSet" + } + ] + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_crstatus{backupplan=~\"$BackupPlan\",cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "ContinuousRestore Info", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "consistentset_count", + "continuousrestoreinstance", + "continuousrestoreplan", + "cr_status" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [ + "logging" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "hide": 2, + "includeAll": false, + "label": "loki", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backupplan_info{cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "hide": 0, + "includeAll": false, + "label": "BackupPlan", + "multi": false, + "name": "BackupPlan", + "options": [], + "query": { + "query": "trilio_backupplan_info{cluster=~\"$Cluster\",kind=\"BackupPlan\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*backupplan=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"BackupPlan\", transaction_resource_name=~\"$BackupPlan\"},transaction_resource_namespace)", + "description": "Backup Namespace", + "hide": 0, + "includeAll": true, + "label": "Backup Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=\"BackupPlan\", transaction_resource_name=~\"$BackupPlan\"},transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"BackupPlan\",transaction_resource_name=~\"$BackupPlan\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"BackupPlan\",transaction_resource_name=~\"$BackupPlan\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "BackupPlan Detail", + "uid": "BackupPlan", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-overview.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-overview.json new file mode 100644 index 000000000..c780e0e67 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backupplan-overview.json @@ -0,0 +1,883 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "grafana-piechart-panel", + "name": "Pie Chart (old)", + "version": "1.6.2" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:13226", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12604, + "graphTooltip": 0, + "id": null, + "iteration": 1655400668324, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Backup Plan Overview

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(46, 122, 122)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 2, + "y": 2 + }, + "id": 34, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Available$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "count(trilio_backupplan_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "title": "All", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(105, 191, 145)", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 6, + "y": 2 + }, + "id": 35, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "Helm", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backupplan_info{applicationtype=\"Helm\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (applicationtype)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{applicationtype}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Helm", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(105, 191, 145)", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 10, + "y": 2 + }, + "id": 36, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Operator$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backupplan_info{applicationtype=\"Operator\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (applicationtype)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{applicationtype}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Operator", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(105, 191, 145)", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 14, + "y": 2 + }, + "id": 37, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Custom$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backupplan_info{applicationtype=\"Custom\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (applicationtype)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{applicationtype}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Custom", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(105, 191, 145)", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 18, + "y": 2 + }, + "id": 40, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Namespace$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_backupplan_info{applicationtype=\"Namespace\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (applicationtype)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{applicationtype}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Namespace", + "transparent": true, + "type": "stat" + }, + { + "aliasColors": {}, + "breakPoint": "50%", + "combine": { + "label": "Others", + "threshold": 0 + }, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fontSize": "80%", + "format": "short", + "gridPos": { + "h": 8, + "w": 7, + "x": 0, + "y": 5 + }, + "id": 39, + "legend": { + "header": "Protected", + "percentage": false, + "show": true, + "values": true + }, + "legendType": "Right side", + "links": [], + "maxDataPoints": 1, + "nullPointMode": "connected", + "pieType": "pie", + "strokeWidth": 1, + "targets": [ + { + "expr": "count(trilio_backupplan_info{applicationtype=~\"$ApplicationType\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (protected)", + "instant": true, + "interval": "", + "legendFormat": "{{protected}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Protected Backup Plan", + "transparent": true, + "type": "grafana-piechart-panel", + "valueName": "current" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "backupplan" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Plan" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Backup Plan Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-BackupPlan=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "applicationtype" + }, + "properties": [ + { + "id": "displayName", + "value": "Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup_count" + }, + "properties": [ + { + "id": "displayName", + "value": "Num of Backups" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "protected" + }, + "properties": [ + { + "id": "displayName", + "value": "Protected" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "lastprotected" + }, + "properties": [ + { + "id": "displayName", + "value": "Last Protected" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "target" + }, + "properties": [ + { + "id": "displayName", + "value": "Target" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Target Detail", + "url": "/d/TargetDetail/target-detail?refresh=5s&var-Target=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "kind" + }, + "properties": [ + { + "id": "displayName", + "value": "BackupPlan Kind" + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 17, + "x": 7, + "y": 5 + }, + "id": 32, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "trilio_backupplan_info{applicationtype=~\"$ApplicationType\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "applicationtype", + "backup_count", + "backupplan", + "kind", + "target", + "lastprotected", + "protected" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 0, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 1, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backupplan_info", + "hide": 0, + "includeAll": true, + "label": "Application Type", + "multi": false, + "name": "ApplicationType", + "options": [], + "query": { + "query": "trilio_backupplan_info", + "refId": "Prometheus-ApplicationType-Variable-Query" + }, + "refresh": 1, + "regex": "/.*applicationtype=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "BackupPlan Overview", + "uid": "BackupPlanOverview", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackup-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackup-detail.json new file mode 100644 index 000000000..3603eb6bd --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackup-detail.json @@ -0,0 +1,820 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "panel", + "id": "gauge", + "name": "Gauge", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:20", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12601, + "graphTooltip": 0, + "id": null, + "iteration": 1655446898146, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 31, + "options": { + "content": "

    Cluster Backup Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "max": 100, + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "dark-red", + "value": null + }, + { + "color": "rgb(255, 255, 255)", + "value": 1 + }, + { + "color": "dark-green", + "value": 100 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 7, + "x": 0, + "y": 2 + }, + "id": 45, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": false + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_status_percentage{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "gauge" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 8, + "y": 2 + }, + "id": 50, + "links": [], + "maxDataPoints": 100, + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backup=~\"$Backup\",namespace=~\"$Namespace\",cluster=~\"$Cluster\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "Cluster Backup Logs", + "gridPos": { + "h": 16, + "w": 12, + "x": 12, + "y": 2 + }, + "id": 52, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"ClusterBackup\",transaction_resource_name=\"$Backup\",service_type=~\"$service_type\",child_transaction_resource_namespace=\"$Namespace\"}", + "refId": "A" + } + ], + "title": "Cluster Backup Logs", + "type": "logs" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 5, + "x": 7, + "y": 3 + }, + "id": 46, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "value" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 6 + }, + "id": 47, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Name", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 6 + }, + "id": 36, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "center", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backupplan$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Cluster Backup Plan", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "custom": { + "align": "left", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "First" + }, + "properties": [ + { + "id": "displayName", + "value": "Value" + } + ] + } + ] + }, + "gridPos": { + "h": 9, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 49, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{ backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Details", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Time", + "applicationtype", + "backup_type", + "completion_ts", + "hook", + "size", + "start_ts", + "target", + "resource_namespace" + ] + } + } + }, + { + "id": "reduce", + "options": { + "reducers": [ + "first" + ] + } + } + ], + "type": "table" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "description": "loki datasource", + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"ClusterBackup\"}", + "hide": 0, + "includeAll": false, + "label": "Cluster Backup", + "multi": false, + "name": "Backup", + "options": [], + "query": { + "query": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"ClusterBackup\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*backup=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterBackup\",transaction_name=~\"$Backup\"},child_transaction_resource_namespace)", + "description": "Child Transaction Namespace", + "hide": 0, + "includeAll": true, + "label": "Backup Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=\"ClusterBackup\",transaction_name=~\"$Backup\"},child_transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterBackup\"},service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"ClusterBackup\"},service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Cluster Backup Detail", + "uid": "ClusterBackup", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackupplan-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackupplan-detail.json new file mode 100644 index 000000000..19b96b0a0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterbackupplan-detail.json @@ -0,0 +1,1234 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:4254", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12605, + "graphTooltip": 0, + "id": null, + "iteration": 1655445890261, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Backup Plan Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 5, + "x": 0, + "y": 2 + }, + "id": 16, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "Status", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 5, + "y": 2 + }, + "id": 9, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^applicationtype$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Application Type", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "False" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 9, + "y": 2 + }, + "id": 10, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^protected$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Protected", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "0" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 13, + "y": 2 + }, + "id": 13, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup_count$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Num of Backups", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 17, + "y": 2 + }, + "id": 14, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^target$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_info{ backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Target", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": true, + "minWidth": 100 + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "backup" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Backup Detail", + "url": "/d/${__data.fields.kind}?refresh=5s&var-Backup=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "completion_ts" + }, + "properties": [ + { + "id": "displayName", + "value": "Completion" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "applicationtype" + }, + "properties": [ + { + "id": "displayName", + "value": "Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup_type" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "kind" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Kind" + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 5 + }, + "id": 12, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backup_info{backupplan=~\"$BackupPlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterBackup\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "Backups", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "backup", + "backup_type", + "completion_ts", + "kind", + "size", + "status", + "target", + "applicationtype" + ] + } + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "BackupPlan Logs", + "gridPos": { + "h": 14, + "w": 12, + "x": 12, + "y": 5 + }, + "id": 18, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"ClusterBackupPlan\",transaction_resource_name=~\"$BackupPlan\",service_type=~\"$service_type\"}", + "refId": "A" + } + ], + "title": "BackupPlan Logs", + "type": "logs" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "consistentset_count" + }, + "properties": [ + { + "id": "displayName", + "value": "ConsistentSet" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "links", + "value": [ + { + "targetBlank": true, + "title": "ConsistentSet Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-ConsistentSet=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "continuousrestoreinstance" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestore Instance" + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "continuousrestoreplan" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestorePlan" + }, + { + "id": "custom.align" + }, + { + "id": "links", + "value": [ + { + "title": "ContinuousRestorePlan Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-ContinuousRestorePlan=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cr_status" + }, + "properties": [ + { + "id": "displayName", + "value": "ContinuousRestore Status" + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ContinuousRestore Instance" + }, + "properties": [ + { + "id": "custom.width", + "value": 200 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ConsistentSet" + }, + "properties": [ + { + "id": "custom.width", + "value": 112 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "ContinuousRestorePlan" + }, + "properties": [ + { + "id": "custom.width", + "value": 177 + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 12 + }, + "id": 20, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [ + { + "desc": false, + "displayName": "ConsistentSet" + } + ] + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_backupplan_crstatus{backupplan=~\"$ClusterBackupPlan\",cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "ContinuousRestore Info", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "consistentset_count", + "continuousrestoreinstance", + "continuousrestoreplan", + "cr_status" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "30s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "hide": 2, + "includeAll": false, + "label": "loki", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_backupplan_info{cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "hide": 0, + "includeAll": false, + "label": "Cluster Backup Plan", + "multi": false, + "name": "BackupPlan", + "options": [], + "query": { + "query": "trilio_backupplan_info{cluster=~\"$Cluster\",kind=\"ClusterBackupPlan\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*backupplan=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterBackupPlan\", transaction_resource_name=~\"$BackupPlan\"},child_transaction_resource_namespace)", + "description": "Backup Namespace", + "hide": 0, + "includeAll": true, + "label": "Backup Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=\"ClusterBackupPlan\", transaction_resource_name=~\"$BackupPlan\"},child_transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterBackupPlan\",transaction_resource_name=~\"$BackupPlan\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"ClusterBackupPlan\",transaction_resource_name=~\"$BackupPlan\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "ClusterBackupPlan Detail", + "uid": "ClusterBackupPlan", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterrestore-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterrestore-detail.json new file mode 100644 index 000000000..ce6851fd1 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/clusterrestore-detail.json @@ -0,0 +1,802 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "panel", + "id": "gauge", + "name": "Gauge", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:1512", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12603, + "graphTooltip": 0, + "id": null, + "iteration": 1655447850637, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Restore Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "max": 100, + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "dark-red", + "value": null + }, + { + "color": "rgb(255, 255, 255)", + "value": 1 + }, + { + "color": "dark-green", + "value": 100 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 7, + "x": 0, + "y": 2 + }, + "id": 13, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": false + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_status_percentage{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "gauge" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 7, + "y": 2 + }, + "id": 17, + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "gridPos": { + "h": 15, + "w": 12, + "x": 12, + "y": 2 + }, + "id": 22, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"ClusterRestore\",transaction_resource_name=~\"$Restore\",service_type=~\"$service_type\",child_transaction_resource_namespace=~\"$Namespace\"}", + "refId": "A" + } + ], + "title": "Restore Logs", + "type": "logs" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Completed" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 5, + "x": 7, + "y": 3 + }, + "id": 15, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "value" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 6 + }, + "id": 6, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^restore$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Restore", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 6 + }, + "id": 20, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Backup", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "custom": { + "align": "left", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "First" + }, + "properties": [ + { + "id": "displayName", + "value": "Value" + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 19, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Details", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Time", + "completion_ts", + "size", + "start_ts", + "target", + "resource_namespace" + ] + } + } + }, + { + "id": "reduce", + "options": { + "reducers": [ + "first" + ] + } + } + ], + "type": "table" + } + ], + "refresh": "30s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "hide": 2, + "includeAll": false, + "label": "loki", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allValue": " ", + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_restore_info{cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "hide": 0, + "includeAll": false, + "label": "Restore", + "multi": false, + "name": "Restore", + "options": [], + "query": { + "query": "trilio_restore_info{cluster=~\"$Cluster\",kind=\"ClusterRestore\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*restore=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterRestore\",transaction_resource_name=~\"$Restore\"},child_transaction_resource_namespace)", + "description": "Restore Namespace", + "hide": 0, + "includeAll": true, + "label": "Restore Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=\"ClusterRestore\",transaction_resource_name=~\"$Restore\"},child_transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ClusterRestore\",transaction_resource_name=~\"$Restore\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"ClusterRestore\",transaction_resource_name=~\"$Restore\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Cluster Restore Detail", + "uid": "ClusterRestore", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/consistentset-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/consistentset-detail.json new file mode 100644 index 000000000..c7bee36c4 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/consistentset-detail.json @@ -0,0 +1,832 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "panel", + "id": "gauge", + "name": "Gauge", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:20", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12601, + "graphTooltip": 0, + "id": null, + "iteration": 1672672013297, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 31, + "options": { + "content": "

    ConsistentSet Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "max": 100, + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "dark-red", + "value": null + }, + { + "color": "rgb(255, 255, 255)", + "value": 1 + }, + { + "color": "dark-green", + "value": 100 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 7, + "x": 0, + "y": 2 + }, + "id": 45, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": false + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_status_percentage{consistentset=~\"$ConsistentSet\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "gauge" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 8, + "y": 2 + }, + "id": 50, + "links": [], + "maxDataPoints": 100, + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{consistentset=~\"$ConsistentSet\",cluster=~\"$Cluster\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "ConsistentSet Logs", + "gridPos": { + "h": 16, + "w": 12, + "x": 12, + "y": 2 + }, + "id": 52, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{child_transaction_resource_name=\"$ConsistentSet\",child_transaction_type=\"ConsistentSet\",transaction_resource_name=\"$ContinuousRestorePlan\",transaction_type=\"ContinuousRestorePlan\"}", + "refId": "A" + } + ], + "title": "ConsistentSet Logs", + "type": "logs" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 5, + "x": 7, + "y": 3 + }, + "id": 46, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "value" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{consistentset=~\"$ConsistentSet\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 6 + }, + "id": 47, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^consistentset$/", + "values": false + }, + "text": { + "valueSize": 10 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{ consistentset=~\"$ConsistentSet\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Name", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 6 + }, + "id": 36, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^continuousrestoreplan$/", + "values": false + }, + "text": { + "valueSize": 10 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{consistentset=~\"$ConsistentSet\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "ContinuousRestorePlan", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "custom": { + "align": "left", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "First" + }, + "properties": [ + { + "id": "displayName", + "value": "Value" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Field" + }, + "properties": [ + { + "id": "displayName" + } + ] + } + ] + }, + "gridPos": { + "h": 9, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 49, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{ consistentset=~\"$ConsistentSet\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Details", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Time", + "completion_ts", + "size", + "start_ts", + "sourcebackupplan", + "sourceinstanceinfo", + "backupName", + "backupNamespace", + "backupStatus" + ] + } + } + }, + { + "id": "reduce", + "options": { + "reducers": [ + "first" + ] + } + } + ], + "type": "table" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "description": "loki datasource", + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 0, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({child_transaction_resource_name=\"$ConsistentSet\",child_transaction_type=\"ConsistentSet\",transaction_resource_name=\"$ContinuousRestorePlan\",transaction_type=\"ContinuousRestorePlan\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({child_transaction_resource_name=\"$ConsistentSet\",child_transaction_type=\"ConsistentSet\",transaction_resource_name=\"$ContinuousRestorePlan\",transaction_type=\"ContinuousRestorePlan\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_consistentset_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"ConsistentSet\"}", + "hide": 0, + "includeAll": false, + "label": "ConsistentSet", + "multi": false, + "name": "ConsistentSet", + "options": [], + "query": { + "query": "trilio_consistentset_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\",kind=\"ConsistentSet\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*consistentset=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_continuousrestoreplan_info{cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "hide": 0, + "includeAll": false, + "label": "ContinuousRestorePlan", + "multi": false, + "name": "ContinuousRestorePlan", + "options": [], + "query": { + "query": "trilio_continuousrestoreplan_info{cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 1, + "regex": "/.*continuousrestoreplan=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "ConsistentSet Detail", + "uid": "ConsistentSet", + "version": 1 +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/continuousrestoreplan-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/continuousrestoreplan-detail.json new file mode 100644 index 000000000..02e201867 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/continuousrestoreplan-detail.json @@ -0,0 +1,1052 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:4254", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12605, + "graphTooltip": 0, + "id": null, + "iteration": 1672673323255, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    ContinuousRestorePlan Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Available" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 0, + "y": 2 + }, + "id": 16, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "Status", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 3, + "y": 2 + }, + "id": 19, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^sourcebackupplan$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{ continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Source BackupPlan", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 7, + "y": 2 + }, + "id": 21, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^sourceinstanceinfo$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{ continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Source Instance", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "0" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 11, + "y": 2 + }, + "id": 13, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^consistentsetcount$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{ continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Num of ConsistentSets", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 5, + "x": 14, + "y": 2 + }, + "id": 20, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^continuousrestorepolicy$/", + "values": false + }, + "text": { + "valueSize": 10 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{ continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "ContinuousRestorePolicy", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 5, + "x": 19, + "y": 2 + }, + "id": 14, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^target$/", + "values": false + }, + "text": { + "valueSize": 10 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_continuousrestoreplan_info{ continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Target", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "consistentset" + }, + "properties": [ + { + "id": "displayName", + "value": "ConsistentSet" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "ConsistentSet Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-ConsistentSet=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "consistentsetscope" + }, + "properties": [ + { + "id": "displayName", + "value": "ConsistentSet Scope" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align", + "value": "left" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "sourcebackupplan" + }, + "properties": [ + { + "id": "displayName", + "value": "Source BackupPlan" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align", + "value": "left" + } + ] + } + ] + }, + "gridPos": { + "h": 12, + "w": 12, + "x": 0, + "y": 5 + }, + "id": 12, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [] + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_consistentset_info{continuousrestoreplan=~\"$ContinuousRestorePlan\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"ConsistentSet\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "title": "ConsistentSets", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "consistentset", + "consistentsetscope", + "size", + "sourcebackupplan", + "status", + "kind" + ] + } + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "ContinuousRestorePlan Logs", + "gridPos": { + "h": 12, + "w": 12, + "x": 12, + "y": 5 + }, + "id": 18, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"ContinuousRestorePlan\",transaction_resource_name=~\"$ContinuousRestorePlan\",service_type=~\"$service_type\"}", + "refId": "A" + } + ], + "title": "ContinuousRestorePlan Logs", + "type": "logs" + } + ], + "refresh": "10s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "hide": 2, + "includeAll": false, + "label": "loki", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 0, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"ContinuousRestorePlan\",transaction_resource_name=~\"$ContinuousRestorePlan\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"ContinuousRestorePlan\",transaction_resource_name=~\"$ContinuousRestorePlan\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_continuousrestoreplan_info{cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "hide": 0, + "includeAll": false, + "label": "ContinuousRestorePlan", + "multi": false, + "name": "ContinuousRestorePlan", + "options": [], + "query": { + "query": "trilio_continuousrestoreplan_info{cluster=~\"$Cluster\",kind=\"ContinuousRestorePlan\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*continuousrestoreplan=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "ContinuousRestorePlan Detail", + "uid": "ContinuousRestorePlan", + "version": 1 +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/logging-dashboard.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/logging-dashboard.json new file mode 100644 index 000000000..3eaae8683 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/logging-dashboard.json @@ -0,0 +1,212 @@ +{ + "__inputs": [ + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "description": "dashboard for logging", + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12611, + "graphTooltip": 0, + "id": null, + "iteration": 1655397917396, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "description": "Live logs is a like 'tail -f' in a real time", + "gridPos": { + "h": 19, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 2, + "options": { + "dedupStrategy": "exact", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{service_type=~\"$service_type\"}", + "queryType": "range", + "refId": "A" + } + ], + "title": "Live logs", + "type": "logs" + } + ], + "refresh": "", + "schemaVersion": 36, + "style": "dark", + "tags": [ + "logging" + ], + "templating": { + "list": [ + { + "current": { + "selected": true, + "text": "All", + "value": "$__all" + }, + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [ + { + "selected": true, + "text": "All", + "value": "$__all" + }, + { + "selected": false, + "text": "ControlPlane", + "value": "ControlPlane" + }, + { + "selected": false, + "text": "WebhookServer", + "value": "WebhookServer" + }, + { + "selected": false, + "text": "Exporter", + "value": "Exporter" + }, + { + "selected": false, + "text": "ResourceCleaner", + "value": "ResourceCleaner" + }, + { + "selected": false, + "text": "WebBackend", + "value": "WebBackend" + }, + { + "selected": false, + "text": "Analyzer", + "value": "Analyzer" + }, + { + "selected": false, + "text": "DexInit", + "value": "DexInit" + } + ], + "query": "ControlPlane,WebhookServer,Exporter,ResourceCleaner,WebBackend,Analyzer,DexInit", + "queryValue": "", + "skipUrlSync": false, + "type": "custom" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "description": "loki datasource", + "hide": 2, + "includeAll": false, + "label": "loki datasource", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + } + ] + }, + "time": { + "from": "now-30m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Logging Dashboard", + "uid": "logging", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/metadata-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/metadata-detail.json new file mode 100644 index 000000000..02a6fb1cc --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/metadata-detail.json @@ -0,0 +1,889 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "7.2.1" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table-old", + "name": "Table (old)", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "7.1.0" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:20", + "builtIn": 1, + "datasource": "-- Grafana --", + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "type": "dashboard" + } + ] + }, + "editable": true, + "gnetId": 12607, + "graphTooltip": 0, + "id": null, + "iteration": 1617300281944, + "links": [], + "panels": [ + { + "content": "

    Backup Metatdata

    ", + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {} + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 12, + "x": 0, + "y": 0 + }, + "id": 58, + "mode": "html", + "options": { + "content": "

    Backup Metatdata

    ", + "mode": "html" + }, + "pluginVersion": "7.1.0", + "timeFrom": null, + "timeShift": null, + "title": "", + "transparent": true, + "type": "text" + }, + { + "content": "

    Restore Metatdata

    ", + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {} + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 12, + "x": 12, + "y": 0 + }, + "id": 59, + "mode": "html", + "options": { + "content": "

    Restore Metatdata

    ", + "mode": "html" + }, + "pluginVersion": "7.1.0", + "timeFrom": null, + "timeShift": null, + "title": "", + "transparent": true, + "type": "text" + }, + { + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {}, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(191, 194, 191)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 2, + "y": 2 + }, + "id": 53, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "7.2.1", + "targets": [ + { + "expr": "count(avg(trilio_backup_metadata_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objecttype))", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Total Component Type", + "type": "stat" + }, + { + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {}, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(191, 194, 191)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 6, + "y": 2 + }, + "id": 54, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "7.2.1", + "targets": [ + { + "expr": "count(avg(trilio_backup_metadata_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objectname))", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Total Components", + "type": "stat" + }, + { + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {}, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(191, 194, 191)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 14, + "y": 2 + }, + "id": 55, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "7.2.1", + "targets": [ + { + "expr": "count(avg(trilio_restore_metadata_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objecttype))", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Total Component Type", + "type": "stat" + }, + { + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {}, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(191, 194, 191)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 18, + "y": 2 + }, + "id": 56, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "7.2.1", + "targets": [ + { + "expr": "count(avg(trilio_restore_metadata_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objectname))", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Total Components", + "type": "stat" + }, + { + "columns": [], + "datasource": "${DS_PROMETHEUS}", + "description": "", + "fieldConfig": { + "defaults": { + "custom": {} + }, + "overrides": [] + }, + "fontSize": "100%", + "gridPos": { + "h": 15, + "w": 12, + "x": 0, + "y": 5 + }, + "id": 42, + "links": [], + "pageSize": null, + "showHeader": true, + "sort": { + "col": 18, + "desc": true + }, + "styles": [ + { + "$$hashKey": "object:10447", + "alias": "Object Type", + "align": "auto", + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "link": false, + "linkTooltip": "Show Metadata Details", + "linkUrl": "/d/0aiPMQMGk/metadata-detail?refresh=5s&var-Backup=${Backup}&var-ObjectType=${__cell}", + "mappingType": 1, + "pattern": "objecttype", + "type": "string" + }, + { + "$$hashKey": "object:1072", + "alias": "Source", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "applicationtype", + "thresholds": [], + "type": "string", + "unit": "short" + }, + { + "$$hashKey": "object:1249", + "alias": "API Version", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": null, + "mappingType": 1, + "pattern": "apiversion", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:3063", + "alias": "Object Name", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "objectname", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:3158", + "alias": "", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:10448", + "alias": "", + "align": "right", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "decimals": 2, + "pattern": "/.*/", + "thresholds": [], + "type": "hidden", + "unit": "short" + } + ], + "targets": [ + { + "expr": "avg(trilio_backup_metadata_info{backup=~\"$Backup\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objectname,objecttype, apiversion)", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Metadata Info", + "transform": "table", + "type": "table-old" + }, + { + "columns": [], + "datasource": "${DS_PROMETHEUS}", + "description": "", + "fieldConfig": { + "defaults": { + "custom": {} + }, + "overrides": [] + }, + "fontSize": "100%", + "gridPos": { + "h": 15, + "w": 12, + "x": 12, + "y": 5 + }, + "id": 60, + "links": [], + "pageSize": null, + "showHeader": true, + "sort": { + "col": 18, + "desc": true + }, + "styles": [ + { + "$$hashKey": "object:10447", + "alias": "Object Type", + "align": "auto", + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "link": false, + "linkTooltip": "Show Metadata Details", + "linkUrl": "/d/0aiPMQMGk/metadata-detail?refresh=5s&var-Backup=${Backup}&var-ObjectType=${__cell}", + "mappingType": 1, + "pattern": "objecttype", + "type": "string" + }, + { + "$$hashKey": "object:1072", + "alias": "Source", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "applicationtype", + "thresholds": [], + "type": "string", + "unit": "short" + }, + { + "$$hashKey": "object:1249", + "alias": "API Version", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": null, + "mappingType": 1, + "pattern": "apiversion", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:3063", + "alias": "Object Name", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "objectname", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:3158", + "alias": "", + "align": "auto", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:10448", + "alias": "", + "align": "right", + "colorMode": null, + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "decimals": 2, + "pattern": "/.*/", + "thresholds": [], + "type": "hidden", + "unit": "short" + } + ], + "targets": [ + { + "expr": "avg(trilio_restore_metadata_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (objectname,objecttype, apiversion)", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "Metadata Info", + "transform": "table", + "type": "table-old" + }, + { + "cacheTimeout": null, + "content": "", + "datasource": "${DS_PROMETHEUS}", + "fieldConfig": { + "defaults": { + "custom": {}, + "mappings": [ + { + "from": "", + "id": 0, + "operator": "", + "text": "Available", + "to": "", + "type": 1, + "value": "1" + }, + { + "from": "", + "id": 1, + "operator": "", + "text": "InProgress", + "to": "", + "type": 1, + "value": "0" + }, + { + "from": "", + "id": 2, + "operator": "", + "text": "Failed", + "to": "", + "type": 1, + "value": "-1" + }, + { + "from": "", + "id": 3, + "operator": "", + "text": "UnKnown", + "to": "", + "type": 1, + "value": "-2" + } + ], + "nullValueMode": "connected", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 8, + "y": 20 + }, + "id": 50, + "interval": null, + "links": [], + "maxDataPoints": 100, + "mode": "markdown", + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "7.1.0", + "targets": [ + { + "expr": "trilio_backup_info{backup=~\"$Backup\",namespace=~\"$Namespace\",cluster=~\"$Cluster\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "timeFrom": null, + "timeShift": null, + "title": "", + "transparent": true, + "type": "text" + } + ], + "refresh": "5s", + "schemaVersion": 26, + "style": "dark", + "tags": [ + "logging" + ], + "templating": { + "list": [ + { + "hide": 2, + "label": "datasource", + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "type": "datasource" + }, + { + "allValue": null, + "current": {}, + "datasource": "${DS_PROMETHEUS}", + "definition": "trilio_system_info", + "hide": 0, + "includeAll": false, + "label": null, + "multi": false, + "name": "Cluster", + "options": [], + "query": "trilio_system_info", + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allValue": null, + "current": {}, + "datasource": "${DS_PROMETHEUS}", + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "label": null, + "multi": false, + "name": "Scope", + "options": [], + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allValue": null, + "current": {}, + "datasource": "${DS_PROMETHEUS}", + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refresh": 1, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allValue": null, + "current": {}, + "datasource": "${DS_PROMETHEUS}", + "definition": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\"}", + "hide": 0, + "includeAll": false, + "label": "Backup", + "multi": false, + "name": "Backup", + "options": [], + "query": "trilio_backup_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\"}", + "refresh": 1, + "regex": "/.*backup=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allValue": null, + "current": {}, + "datasource": "${DS_PROMETHEUS}", + "definition": "trilio_restore_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\"}", + "hide": 0, + "includeAll": false, + "label": "Restore", + "multi": false, + "name": "Restore", + "options": [], + "query": "trilio_restore_info{cluster=~\"$Cluster\",install_namespace=~\"$Install_Namespace\"}", + "refresh": 1, + "regex": "/.*restore=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Metadata Detail", + "uid": "Metadata", + "version": 1 +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/overview.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/overview.json new file mode 100644 index 000000000..785f63bdd --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/overview.json @@ -0,0 +1,1429 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "piechart", + "name": "Pie chart", + "version": "" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:4047", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12599, + "graphTooltip": 0, + "id": null, + "iteration": 1672675438573, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 5, + "w": 10, + "x": 0, + "y": 0 + }, + "id": 45, + "options": { + "content": "

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "mappings": [ + { + "options": { + "0": { + "color": "red", + "index": 1, + "text": "Not Ready" + }, + "1": { + "color": "green", + "index": 0, + "text": "Ready" + } + }, + "type": "value" + } + ] + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 7, + "x": 11, + "y": 0 + }, + "id": 47, + "options": { + "legend": { + "displayMode": "list", + "placement": "bottom" + }, + "pieType": "donut", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_component_status", + "instant": true, + "interval": "", + "legendFormat": "{{deployment}}-{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "piechart" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 3, + "x": 18, + "y": 0 + }, + "id": 42, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^tvk_version$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_system_info{ install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "TVK Version", + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 3, + "x": 21, + "y": 0 + }, + "id": 43, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^scope$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_system_info{ install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "TVK Scope", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "decimals": 0, + "mappings": [], + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Available" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#37872D", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#C4162A", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "InProgress" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#FADE2A", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "UnKnown" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "rgb(43, 36, 36)", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 0, + "y": 5 + }, + "id": 34, + "links": [ + { + "title": "Show Backup Overview", + "url": "/d/BackupOverview/backup-overview?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ], + "maxDataPoints": 1, + "options": { + "legend": { + "calcs": [], + "displayMode": "table", + "placement": "bottom", + "values": [ + "value" + ] + }, + "pieType": "donut", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "count(trilio_backup_info{ install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "title": "Backup Summary", + "type": "piechart" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "decimals": 0, + "mappings": [], + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#37872D", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 8, + "y": 5 + }, + "id": 36, + "links": [ + { + "title": "Show Backup Plan Overview", + "url": "/d/BackupPlanOverview/backupplan-overview?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ], + "maxDataPoints": 1, + "options": { + "legend": { + "calcs": [], + "displayMode": "table", + "placement": "bottom", + "values": [ + "value" + ] + }, + "pieType": "donut", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "count(trilio_backupplan_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (kind) or count(trilio_continuousrestoreplan_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (kind)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "__auto", + "refId": "A" + } + ], + "title": " Plan Summary ", + "type": "piechart" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "left", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Total Targets" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "custom.align", + "value": "left" + }, + { + "id": "decimals", + "value": 0 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Health" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "vendorType" + }, + "properties": [ + { + "id": "displayName", + "value": "Vendor Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "eventTarget" + }, + "properties": [ + { + "id": "displayName", + "value": "Target Type" + }, + { + "id": "mappings", + "value": [ + { + "options": { + "false": { + "index": 1, + "text": "Data Target" + }, + "true": { + "index": 0, + "text": "Event Target" + } + }, + "type": "value" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "target" + }, + "properties": [ + { + "id": "displayName", + "value": "Target Name" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Target Type" + }, + "properties": [ + { + "id": "custom.width", + "value": 100 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Health" + }, + "properties": [ + { + "id": "custom.width", + "value": 84 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Target Name" + }, + "properties": [ + { + "id": "custom.width", + "value": 106 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Vendor Type" + }, + "properties": [ + { + "id": "custom.width", + "value": 96 + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 16, + "y": 5 + }, + "id": 40, + "links": [ + { + "title": "Show Target Details", + "url": "/d/TargetDetail/target-detail?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ], + "options": { + "footer": { + "fields": [ + "Value" + ], + "reducer": [ + "sum" + ], + "show": true + }, + "showHeader": true, + "sortBy": [] + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Target Summary", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "status", + "vendorType", + "Value", + "target", + "eventTarget" + ] + } + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "decimals": 0, + "mappings": [], + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Available" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#37872D", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#C4162A", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "InProgress" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#FADE2A", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "UnKnown" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "rgb(43, 36, 36)", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 0, + "y": 13 + }, + "id": 48, + "links": [ + { + "title": "Show ConsistentSet Overview", + "url": "/d/ConsistentSet/consistentset-detail?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ], + "maxDataPoints": 1, + "options": { + "legend": { + "calcs": [], + "displayMode": "table", + "placement": "bottom", + "values": [ + "value" + ] + }, + "pieType": "donut", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "count(trilio_consistentset_info{ install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "title": "ConsistentSet Summary", + "type": "piechart" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "decimals": 0, + "mappings": [], + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Completed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#37872D", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#C4162A", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value #C" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "#E0B400", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 8, + "y": 13 + }, + "id": 35, + "links": [ + { + "title": "Show Restore Overview", + "url": "/d/RestoreOverview/restore-overview?refresh=5s&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ], + "maxDataPoints": 1, + "options": { + "legend": { + "calcs": [], + "displayMode": "table", + "placement": "bottom", + "values": [ + "value" + ] + }, + "pieType": "donut", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "count(trilio_restore_info{ install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status) ", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "title": "Restore Summary", + "type": "piechart" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "left", + "displayMode": "auto", + "inspect": false, + "width": 50 + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "deployment" + }, + "properties": [ + { + "id": "displayName", + "value": "Component" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Health" + }, + { + "id": "unit", + "value": "none" + }, + { + "id": "custom.displayMode", + "value": "color-background" + }, + { + "id": "custom.align", + "value": "left" + }, + { + "id": "thresholds", + "value": { + "mode": "absolute", + "steps": [ + { + "color": "#C4162A", + "value": null + }, + { + "color": "#C4162A", + "value": 0 + }, + { + "color": "rgba(50, 172, 45, 0.97)", + "value": 1 + } + ] + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Component" + }, + "properties": [ + { + "id": "custom.width", + "value": 264 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Health" + }, + "properties": [ + { + "id": "custom.width", + "value": 134 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Status" + }, + "properties": [ + { + "id": "custom.width", + "value": 75 + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 8, + "x": 16, + "y": 13 + }, + "id": 32, + "links": [ + { + "targetBlank": false, + "title": "Logging Dashboard", + "url": "/d/logging/logging-dashboard?orgId=1" + } + ], + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [] + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_component_status{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "TVK Health", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "deployment", + "status", + "Value" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 0, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 1, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-5m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Overview", + "uid": "Overview", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-detail.json new file mode 100644 index 000000000..aa3fcad9a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-detail.json @@ -0,0 +1,901 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "panel", + "id": "gauge", + "name": "Gauge", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "table-old", + "name": "Table (old)", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:1512", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12603, + "graphTooltip": 0, + "id": null, + "iteration": 1655448141180, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Restore Detail

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "max": 100, + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "dark-red", + "value": null + }, + { + "color": "rgb(255, 255, 255)", + "value": 1 + }, + { + "color": "dark-green", + "value": 100 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 7, + "x": 0, + "y": 2 + }, + "id": 13, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": false + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_status_percentage{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Restore\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "gauge" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 1, + "w": 4, + "x": 7, + "y": 2 + }, + "id": 17, + "options": { + "content": "", + "mode": "markdown" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "columns": [], + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fontSize": "100%", + "gridPos": { + "h": 7, + "w": 12, + "x": 12, + "y": 2 + }, + "id": 11, + "showHeader": true, + "sort": { + "col": 0, + "desc": true + }, + "styles": [ + { + "$$hashKey": "object:2545", + "alias": "Object Type", + "align": "auto", + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "link": true, + "linkTooltip": "${__cell}", + "linkUrl": "/d/Metadata/metadata-detail?var-Restore=${Restore}&var-ObjectType=${__cell}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}", + "pattern": "objecttype", + "type": "string" + }, + { + "$$hashKey": "object:1086", + "alias": "Source", + "align": "auto", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "decimals": 2, + "mappingType": 1, + "pattern": "applicationtype", + "thresholds": [], + "type": "string", + "unit": "short" + }, + { + "$$hashKey": "object:1112", + "alias": "Count", + "align": "auto", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "dateFormat": "YYYY-MM-DD HH:mm:ss", + "mappingType": 1, + "pattern": "Value", + "thresholds": [], + "type": "number", + "unit": "short" + }, + { + "$$hashKey": "object:2546", + "alias": "", + "align": "right", + "colors": [ + "rgba(245, 54, 54, 0.9)", + "rgba(237, 129, 40, 0.89)", + "rgba(50, 172, 45, 0.97)" + ], + "decimals": 2, + "pattern": "/.*/", + "thresholds": [], + "type": "hidden", + "unit": "short" + } + ], + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "avg(trilio_restore_metadata_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Restore\"}) by (objecttype, applicationtype)", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Metadata Info", + "transform": "table", + "type": "table-old" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "InProgress" + }, + "1": { + "text": "Completed" + }, + "-1": { + "text": "Failed" + }, + "-2": { + "text": "UnKnown" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + }, + { + "color": "dark-red", + "value": -1 + }, + { + "color": "blue", + "value": 0 + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 2, + "w": 5, + "x": 7, + "y": 3 + }, + "id": 15, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "value" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Restore\"}", + "format": "time_series", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{status}}", + "refId": "A" + } + ], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 6 + }, + "id": 6, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^restore$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Restore", + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 6 + }, + "id": 20, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^backup$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Backup", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "custom": { + "align": "left", + "displayMode": "auto", + "filterable": false, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "First" + }, + "properties": [ + { + "id": "displayName", + "value": "Value" + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 19, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": "${DS_PROMETHEUS}", + "expr": "trilio_restore_info{ restore=~\"$Restore\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\",kind=\"Restore\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Details", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Time", + "completion_ts", + "size", + "start_ts", + "target", + "resource_namespace" + ] + } + } + }, + { + "id": "reduce", + "options": { + "reducers": [ + "first" + ] + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 9 + }, + "id": 22, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"Restore\",transaction_resource_name=~\"$Restore\",service_type=~\"$service_type\"}", + "refId": "A" + } + ], + "title": "Restore Logs", + "type": "logs" + } + ], + "refresh": "30s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "hide": 2, + "includeAll": false, + "label": "loki", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_restore_info{cluster=~\"$Cluster\",kind=\"Restore\"}", + "hide": 0, + "includeAll": false, + "label": "Restore", + "multi": false, + "name": "Restore", + "options": [], + "query": { + "query": "trilio_restore_info{cluster=~\"$Cluster\",kind=\"Restore\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*restore=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=~\"Restore\",transaction_resource_name=~\"$Restore\"},transaction_resource_namespace)", + "description": "Restore Namespace", + "hide": 0, + "includeAll": true, + "label": "Restore Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=~\"Restore\",transaction_resource_name=~\"$Restore\"},transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"Restore\",transaction_resource_name=~\"$Restore\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"Restore\",transaction_resource_name=~\"$Restore\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Restore Detail", + "uid": "Restore", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-overview.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-overview.json new file mode 100644 index 000000000..2072f9941 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/restore-overview.json @@ -0,0 +1,853 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:48276", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12602, + "graphTooltip": 0, + "id": null, + "iteration": 1655396733765, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 23, + "x": 1, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Restores Overview

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(115, 181, 181)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 2, + "y": 2 + }, + "id": 35, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_restore_info{install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) ", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "All", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "All", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 6, + "y": 2 + }, + "id": 36, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Completed$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_restore_info{status=\"Completed\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Completed", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "blue", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 10, + "y": 2 + }, + "id": 39, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^InProgress$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_restore_info{status=\"InProgress\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "InProgress", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "red", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 14, + "y": 2 + }, + "id": 38, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "/^Failed$/", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_restore_info{status=\"Failed\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "Failed", + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "links": [], + "mappings": [], + "min": 0, + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "rgb(79, 145, 145)", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 18, + "y": 2 + }, + "id": 37, + "links": [], + "options": { + "colorMode": "background", + "graphMode": "none", + "justifyMode": "center", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "expr": "count(trilio_restore_info{status=\"UnKnown\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (status)", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "{{status}}", + "refId": "A", + "datasource": "${DS_PROMETHEUS}" + } + ], + "title": "UnKnown", + "transformations": [], + "transparent": true, + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "restore" + }, + "properties": [ + { + "id": "displayName", + "value": "Restore" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Show Restore Detail", + "url": "/d/${__data.fields.kind}/?refresh=5s&var-Restore=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "completion_ts" + }, + "properties": [ + { + "id": "displayName", + "value": "Completion" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "unit", + "value": "time: YYYY-MM-DD HH:mm:ss.SSS" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Value" + }, + "properties": [ + { + "id": "displayName", + "value": "Percentage" + }, + { + "id": "unit", + "value": "percent" + }, + { + "id": "custom.align" + }, + { + "id": "decimals", + "value": 0 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "target" + }, + "properties": [ + { + "id": "displayName", + "value": "Target" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Show Target Detail", + "url": "/d/TargetDetail/target-detail?refresh=5s&var-Target=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "kind" + }, + "properties": [ + { + "id": "displayName", + "value": "Restore Kind" + } + ] + } + ] + }, + "gridPos": { + "h": 10, + "w": 24, + "x": 0, + "y": 5 + }, + "id": 29, + "options": { + "footer": { + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "expr": "trilio_restore_status_percentage{status=~\"$Status\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "", + "refId": "A" + } + ], + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "completion_ts", + "kind", + "restore", + "size", + "status", + "target", + "Value", + "backup" + ] + } + } + } + ], + "type": "table" + } + ], + "refresh": "5s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 1, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_restore_info", + "hide": 0, + "includeAll": true, + "label": "status", + "multi": false, + "name": "Status", + "options": [], + "query": { + "query": "trilio_restore_info", + "refId": "Prometheus-Status-Variable-Query" + }, + "refresh": 1, + "regex": "/.*status=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Restore Overview", + "uid": "RestoreOverview", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/target-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/target-detail.json new file mode 100644 index 000000000..5fa45353d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/target-detail.json @@ -0,0 +1,1327 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:63480", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12606, + "graphTooltip": 0, + "id": 14, + "iteration": 1655445162139, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 4, + "options": { + "content": "

    Targets

    ", + "mode": "html" + }, + "pluginVersion": "8.5.0", + "transparent": true, + "type": "text" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [ + { + "options": { + "0": { + "text": "Unavailable" + }, + "1": { + "text": "Available" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#d44a3a", + "value": null + }, + { + "color": "dark-red", + "value": 0 + }, + { + "color": "#299c46", + "value": 1 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 0, + "y": 2 + }, + "id": 8, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "avg(trilio_target_info{target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"})", + "format": "time_series", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Health", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "false": { + "index": 1, + "text": "Data Target" + }, + "true": { + "index": 0, + "text": "Event Target" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 3, + "y": 2 + }, + "id": 43, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^eventTarget$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Target Type", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "false": { + "text": "False" + }, + "true": { + "text": "True" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 2, + "x": 7, + "y": 2 + }, + "id": 35, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^browsing$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Browsing", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "text": "N/A" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 9, + "y": 2 + }, + "id": 39, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^threshold_capacity$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Threshold Capacity", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#c7d0d9", + "value": null + } + ] + }, + "unit": "decbytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 12, + "y": 2 + }, + "id": 38, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "value", + "fieldOptions": { + "calcs": [ + "mean" + ] + }, + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "sum" + ], + "fields": "", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "sum(trilio_backup_storage{target=~\"$Target\",status=~\"Available\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"})", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Total Capacity", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 3, + "x": 15, + "y": 2 + }, + "id": 36, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^vendorType$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Vendor Type", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 2, + "x": 18, + "y": 2 + }, + "id": 40, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^vendor$/", + "values": false + }, + "text": { + "valueSize": 30 + }, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Vendor ", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "match": "null", + "result": { + "text": "N/A" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "dateTimeAsIso" + }, + "overrides": [] + }, + "gridPos": { + "h": 3, + "w": 4, + "x": 20, + "y": 2 + }, + "id": 37, + "links": [], + "maxDataPoints": 100, + "options": { + "colorMode": "none", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "horizontal", + "reduceOptions": { + "calcs": [ + "first" + ], + "fields": "/^creation_ts$/", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "trilio_target_info{ target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + } + ], + "title": "Creattion Timestamp", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "displayMode": "auto", + "inspect": false, + "minWidth": 100 + }, + "decimals": 2, + "displayName": "", + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "backup" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "links", + "value": [ + { + "targetBlank": false, + "title": "Show Backup Detail", + "url": "/d/${__data.fields.kind}?refresh=5s&var-Backup=${__value.text}&var-Cluster=${Cluster}&var-Install_Namespace=${Install_Namespace}" + } + ] + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backupplan" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Plan" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "completion_ts" + }, + "properties": [ + { + "id": "displayName", + "value": "Completion " + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "size" + }, + "properties": [ + { + "id": "displayName", + "value": "Size" + }, + { + "id": "unit", + "value": "decbytes" + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "-" + }, + "properties": [ + { + "id": "displayName", + "value": "Average Data Transfer" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "target" + }, + "properties": [ + { + "id": "displayName", + "value": "Target" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "status" + }, + "properties": [ + { + "id": "displayName", + "value": "Status" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "backup_type" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup Type" + }, + { + "id": "unit", + "value": "short" + }, + { + "id": "decimals", + "value": 2 + }, + { + "id": "custom.align" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "kind" + }, + "properties": [ + { + "id": "displayName", + "value": "Backup KInd" + } + ] + } + ] + }, + "gridPos": { + "h": 18, + "w": 12, + "x": 0, + "y": 5 + }, + "id": 32, + "options": { + "footer": { + "enablePagination": false, + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true + }, + "pluginVersion": "8.5.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "exemplar": false, + "expr": "topk(10,trilio_backup_info{target=~\"$Target\",install_namespace=~\"$Install_Namespace\",cluster=~\"$Cluster\"}) by (size)", + "format": "table", + "instant": true, + "interval": "", + "intervalFactor": 1, + "legendFormat": "{{backup}} {{status}}", + "refId": "A" + } + ], + "title": "Backups", + "transformations": [ + { + "id": "merge", + "options": { + "reducers": [] + } + }, + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "backup", + "backup_type", + "backupplan", + "completion_ts", + "kind", + "size", + "status", + "target" + ] + } + } + } + ], + "type": "table" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "gridPos": { + "h": 18, + "w": 12, + "x": 12, + "y": 5 + }, + "id": 42, + "options": { + "dedupStrategy": "none", + "enableLogDetails": true, + "prettifyLogMessage": false, + "showCommonLabels": false, + "showLabels": false, + "showTime": true, + "sortOrder": "Descending", + "wrapLogMessage": false + }, + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "expr": "{transaction_type=\"Target\",transaction_resource_name=~\"$Target\",service_type=~\"$service_type\",transaction_resource_namespace=~\"$Namespace\"}", + "refId": "A" + } + ], + "title": "Target Logs ", + "type": "logs" + } + ], + "refresh": "30s", + "schemaVersion": 36, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "Prometheus" + }, + "description": "prometheus datasource", + "hide": 2, + "includeAll": false, + "label": "datasource", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "Loki" + }, + "description": "loki datasource", + "hide": 2, + "includeAll": false, + "label": "loki datasource", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Cluster", + "options": [], + "query": { + "query": "trilio_system_info", + "refId": "Prometheus-Cluster-Variable-Query" + }, + "refresh": 1, + "regex": "/.*cluster=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{cluster=~\"$Cluster\"}", + "hide": 2, + "includeAll": false, + "multi": false, + "name": "Scope", + "options": [], + "query": { + "query": "trilio_system_info{cluster=~\"$Cluster\"}", + "refId": "Prometheus-Scope-Variable-Query" + }, + "refresh": 1, + "regex": "/.*scope=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Install Namespace", + "multi": false, + "name": "Install_Namespace", + "options": [], + "query": { + "query": "trilio_system_info{scope=~\"$Scope\",cluster=~\"$Cluster\"}", + "refId": "Prometheus-Install_Namespace-Variable-Query" + }, + "refresh": 2, + "regex": "/.*install_namespace=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "definition": "trilio_target_info{cluster=~\"$Cluster\"}", + "hide": 0, + "includeAll": false, + "label": "Target", + "multi": false, + "name": "Target", + "options": [], + "query": { + "query": "trilio_target_info{cluster=~\"$Cluster\"}", + "refId": "StandardVariableQuery" + }, + "refresh": 2, + "regex": "/.*target=\"([^\"]*).*/", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"Target\",transaction_resource_name=~\"$Target\"},transaction_resource_namespace)", + "description": "Namespace", + "hide": 0, + "includeAll": true, + "label": "Namespace", + "multi": false, + "name": "Namespace", + "options": [], + "query": "label_values({transaction_type=\"Target\",transaction_resource_name=~\"$Target\"},transaction_resource_namespace)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "current": {}, + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "definition": "label_values({transaction_type=\"Target\"}, service_type)", + "description": "Service Type", + "hide": 0, + "includeAll": true, + "label": "Service Type", + "multi": false, + "name": "service_type", + "options": [], + "query": "label_values({transaction_type=\"Target\"}, service_type)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timepicker": { + "refresh_intervals": [ + "10s", + "30s", + "1m", + "5m", + "15m", + "30m", + "1h", + "2h", + "1d" + ] + }, + "timezone": "", + "title": "Target Detail", + "uid": "TargetDetail", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/NOTES.txt new file mode 100644 index 000000000..1fc8436d9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/NOTES.txt @@ -0,0 +1,54 @@ +1. Get your '{{ .Values.adminUser }}' user password by running: + + kubectl get secret --namespace {{ template "grafana.namespace" . }} {{ template "grafana.fullname" . }} -o jsonpath="{.data.admin-password}" | base64 --decode ; echo + +2. The Grafana server can be accessed via port {{ .Values.service.port }} on the following DNS name from within your cluster: + + {{ template "grafana.fullname" . }}.{{ template "grafana.namespace" . }}.svc.cluster.local +{{ if .Values.ingress.enabled }} + If you bind grafana to 80, please update values in values.yaml and reinstall: + ``` + securityContext: + runAsUser: 0 + runAsGroup: 0 + fsGroup: 0 + + command: + - "setcap" + - "'cap_net_bind_service=+ep'" + - "/usr/sbin/grafana-server &&" + - "sh" + - "/run.sh" + ``` + Details refer to https://grafana.com/docs/installation/configuration/#http-port. + Or grafana would always crash. + + From outside the cluster, the server URL(s) are: +{{- range .Values.ingress.hosts }} + http://{{ . }} +{{- end }} +{{ else }} + Get the Grafana URL to visit by running these commands in the same shell: +{{ if contains "NodePort" .Values.service.type -}} + export NODE_PORT=$(kubectl get --namespace {{ template "grafana.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "grafana.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ template "grafana.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{ else if contains "LoadBalancer" .Values.service.type -}} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ template "grafana.namespace" . }} -w {{ template "grafana.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ template "grafana.namespace" . }} {{ template "grafana.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + http://$SERVICE_IP:{{ .Values.service.port -}} +{{ else if contains "ClusterIP" .Values.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ template "grafana.namespace" . }} -l "app.kubernetes.io/name={{ template "grafana.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ template "grafana.namespace" . }} port-forward $POD_NAME 3000 +{{- end }} +{{- end }} + +3. Login with the password from step 1 and the username: {{ .Values.adminUser }} + +{{- if not .Values.persistence.enabled }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the Grafana pod is terminated. ##### +################################################################################# +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_helpers.tpl new file mode 100644 index 000000000..e3a1fff46 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_helpers.tpl @@ -0,0 +1,165 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "grafana.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 "grafana.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 "grafana.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account +*/}} +{{- define "grafana.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "grafana.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{- define "grafana.serviceAccountNameTest" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (print (include "grafana.fullname" .) "-test") .Values.serviceAccount.nameTest }} +{{- else -}} + {{ default "default" .Values.serviceAccount.nameTest }} +{{- end -}} +{{- end -}} + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts +*/}} +{{- define "grafana.namespace" -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- end -}} +{{- end -}} + +{{/* +Common labels +*/}} +{{- define "grafana.labels" -}} +helm.sh/chart: {{ include "grafana.chart" . }} +{{ include "grafana.selectorLabels" . }} +{{- if or .Chart.AppVersion .Values.image.tag }} +app.kubernetes.io/version: {{ .Values.image.tag | default .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- if .Values.extraLabels }} +{{ toYaml .Values.extraLabels }} +{{- end }} +{{- end -}} + +{{/* +Selector labels +*/}} +{{- define "grafana.selectorLabels" -}} +app.kubernetes.io/name: {{ include "grafana.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/managed-by: k8s-triliovault-operator +app.kubernetes.io/part-of: k8s-triliovault-operator +{{- end -}} + +{{/* +Common labels +*/}} +{{- define "grafana.imageRenderer.labels" -}} +helm.sh/chart: {{ include "grafana.chart" . }} +{{ include "grafana.imageRenderer.selectorLabels" . }} +{{- if or .Chart.AppVersion .Values.image.tag }} +app.kubernetes.io/version: {{ .Values.image.tag | default .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} + +{{/* +Selector labels ImageRenderer +*/}} +{{- define "grafana.imageRenderer.selectorLabels" -}} +app.kubernetes.io/name: {{ include "grafana.name" . }}-image-renderer +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end -}} + +{{/* +Looks if there's an existing secret and reuse its password. If not it generates +new password and use it. +*/}} +{{- define "grafana.password" -}} +{{- $secret := (lookup "v1" "Secret" (include "grafana.namespace" .) (include "grafana.fullname" .) ) -}} + {{- if $secret -}} + {{- index $secret "data" "admin-password" -}} + {{- else -}} + {{- (randAlphaNum 40) | b64enc | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for rbac. +*/}} +{{- define "grafana.rbac.apiVersion" -}} + {{- if .Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1" }} + {{- print "rbac.authorization.k8s.io/v1" -}} + {{- else -}} + {{- print "rbac.authorization.k8s.io/v1beta1" -}} + {{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "grafana.ingress.apiVersion" -}} + {{- if and (.Capabilities.APIVersions.Has "networking.k8s.io/v1") (semverCompare ">= 1.19-0" .Capabilities.KubeVersion.Version) -}} + {{- print "networking.k8s.io/v1" -}} + {{- else if .Capabilities.APIVersions.Has "networking.k8s.io/v1beta1" -}} + {{- print "networking.k8s.io/v1beta1" -}} + {{- else -}} + {{- print "extensions/v1beta1" -}} + {{- end -}} +{{- end -}} + +{{/* +Return if ingress is stable. +*/}} +{{- define "grafana.ingress.isStable" -}} + {{- eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1" -}} +{{- end -}} + +{{/* +Return if ingress supports ingressClassName. +*/}} +{{- define "grafana.ingress.supportsIngressClassName" -}} + {{- or (eq (include "grafana.ingress.isStable" .) "true") (and (eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18-0" .Capabilities.KubeVersion.Version)) -}} +{{- end -}} + +{{/* +Return if ingress supports pathType. +*/}} +{{- define "grafana.ingress.supportsPathType" -}} + {{- or (eq (include "grafana.ingress.isStable" .) "true") (and (eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18-0" .Capabilities.KubeVersion.Version)) -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_pod.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_pod.tpl new file mode 100644 index 000000000..af2c4036e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/_pod.tpl @@ -0,0 +1,748 @@ + +{{- define "grafana.pod" -}} +{{- if .Values.schedulerName }} +schedulerName: "{{ .Values.schedulerName }}" +{{- end }} +serviceAccountName: {{ template "grafana.serviceAccountName" . }} +automountServiceAccountToken: {{ .Values.serviceAccount.autoMount }} +{{- if .Values.securityContext }} +securityContext: +{{ toYaml .Values.securityContext | indent 2 }} +{{- end }} +{{- if .Values.hostAliases }} +hostAliases: +{{ toYaml .Values.hostAliases | indent 2 }} +{{- end }} +{{- if .Values.priorityClassName }} +priorityClassName: {{ .Values.priorityClassName }} +{{- end }} +{{- if ( or .Values.persistence.enabled .Values.dashboards .Values.sidecar.notifiers.enabled .Values.extraInitContainers (and .Values.sidecar.datasources.enabled .Values.sidecar.datasources.initDatasources)) }} +initContainers: +{{- end }} +{{- if ( and .Values.persistence.enabled .Values.initChownData.enabled ) }} + - name: init-chown-data + {{- if .Values.initChownData.image.sha }} + image: "{{ .Values.initChownData.image.registry }}/{{ .Values.initChownData.image.repository }}:{{ .Values.initChownData.image.tag }}@sha256:{{ .Values.initChownData.image.sha }}" + {{- else }} + image: "{{ .Values.initChownData.image.registry }}/{{ .Values.initChownData.image.repository }}:{{ .Values.initChownData.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.initChownData.image.pullPolicy }} + securityContext: + runAsNonRoot: false + runAsUser: 0 + command: ["chown", "-R", "{{ .Values.securityContext.runAsUser }}:{{ .Values.securityContext.runAsGroup }}", "/var/lib/grafana"] + resources: +{{ toYaml .Values.initChownData.resources | indent 6 }} + volumeMounts: + - name: storage + mountPath: "/var/lib/grafana" +{{- if .Values.persistence.subPath }} + subPath: {{ tpl .Values.persistence.subPath . }} +{{- end }} +{{- end }} +{{- if .Values.dashboards }} + - name: download-dashboards + {{- if .Values.downloadDashboardsImage.sha }} + image: "{{ .Values.downloadDashboardsImage.registry }}/{{ .Values.downloadDashboardsImage.repository }}:{{ .Values.downloadDashboardsImage.tag }}@sha256:{{ .Values.downloadDashboardsImage.sha }}" + {{- else }} + image: "{{ .Values.downloadDashboardsImage.registry }}/{{ .Values.downloadDashboardsImage.repository }}:{{ .Values.downloadDashboardsImage.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.downloadDashboardsImage.pullPolicy }} + command: ["/bin/sh"] + args: [ "-c", "mkdir -p /var/lib/grafana/dashboards/default && /bin/sh -x /etc/grafana/download_dashboards.sh" ] + resources: +{{ toYaml .Values.downloadDashboards.resources | indent 6 }} + env: +{{- range $key, $value := .Values.downloadDashboards.env }} + - name: "{{ $key }}" + value: "{{ $value }}" +{{- end }} +{{- if .Values.downloadDashboards.envFromSecret }} + envFrom: + - secretRef: + name: {{ tpl .Values.downloadDashboards.envFromSecret . }} +{{- end }} + volumeMounts: + - name: config + mountPath: "/etc/grafana/download_dashboards.sh" + subPath: download_dashboards.sh + - name: storage + mountPath: "/var/lib/grafana" +{{- if .Values.persistence.subPath }} + subPath: {{ tpl .Values.persistence.subPath . }} +{{- end }} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + {{- end }} +{{- end }} +{{- if and .Values.sidecar.datasources.enabled .Values.sidecar.datasources.initDatasources }} + - name: {{ template "grafana.name" . }}-init-sc-datasources + {{- if .Values.sidecar.image.sha }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + - name: METHOD + value: "LIST" + - name: LABEL + value: "{{ .Values.sidecar.datasources.label }}" + {{- if .Values.sidecar.datasources.labelValue }} + - name: LABEL_VALUE + value: {{ quote .Values.sidecar.datasources.labelValue }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/datasources" + - name: RESOURCE + value: {{ quote .Values.sidecar.datasources.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- if .Values.sidecar.datasources.searchNamespace }} + - name: NAMESPACE + value: "{{ .Values.sidecar.datasources.searchNamespace | join "," }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} + resources: +{{ toYaml .Values.sidecar.resources | indent 6 }} +{{- if .Values.sidecar.securityContext }} + securityContext: +{{- toYaml .Values.sidecar.securityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" +{{- end }} +{{- if .Values.sidecar.notifiers.enabled }} + - name: {{ template "grafana.name" . }}-sc-notifiers + {{- if .Values.sidecar.image.sha }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + - name: METHOD + value: LIST + - name: LABEL + value: "{{ .Values.sidecar.notifiers.label }}" + - name: FOLDER + value: "/etc/grafana/provisioning/notifiers" + - name: RESOURCE + value: {{ quote .Values.sidecar.notifiers.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- if .Values.sidecar.notifiers.searchNamespace }} + - name: NAMESPACE + value: "{{ .Values.sidecar.notifiers.searchNamespace | join "," }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} +{{- if .Values.sidecar.livenessProbe }} + livenessProbe: +{{ toYaml .Values.livenessProbe | indent 6 }} +{{- end }} +{{- if .Values.sidecar.readinessProbe }} + readinessProbe: +{{ toYaml .Values.readinessProbe | indent 6 }} +{{- end }} + resources: +{{ toYaml .Values.sidecar.resources | indent 6 }} +{{- if .Values.sidecar.securityContext }} + securityContext: +{{- toYaml .Values.sidecar.securityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: sc-notifiers-volume + mountPath: "/etc/grafana/provisioning/notifiers" +{{- end}} +{{- if .Values.extraInitContainers }} +{{ tpl (toYaml .Values.extraInitContainers) . | indent 2 }} +{{- end }} +{{- if .Values.image.pullSecrets }} +imagePullSecrets: +{{- $root := . }} +{{- range .Values.image.pullSecrets }} + - name: {{ tpl . $root }} +{{- end}} +{{- end }} +{{- if not .Values.enableKubeBackwardCompatibility }} +enableServiceLinks: {{ .Values.enableServiceLinks }} +{{- end }} +containers: +{{- if .Values.sidecar.dashboards.enabled }} + - name: {{ template "grafana.name" . }}-sc-dashboard + {{- if .Values.sidecar.image.sha }} + image: "{{ .Values.sidecar.image.registry }}{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + - name: METHOD + value: {{ .Values.sidecar.dashboards.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.dashboards.label }}" + {{- if .Values.sidecar.dashboards.labelValue }} + - name: LABEL_VALUE + value: {{ quote .Values.sidecar.dashboards.labelValue }} + {{- end }} + - name: FOLDER + value: "{{ .Values.sidecar.dashboards.folder }}{{- with .Values.sidecar.dashboards.defaultFolderName }}/{{ . }}{{- end }}" + - name: RESOURCE + value: {{ quote .Values.sidecar.dashboards.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- if .Values.sidecar.dashboards.searchNamespace }} + - name: NAMESPACE + value: "{{ .Values.sidecar.dashboards.searchNamespace | join "," }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} + {{- if .Values.sidecar.dashboards.folderAnnotation }} + - name: FOLDER_ANNOTATION + value: "{{ .Values.sidecar.dashboards.folderAnnotation }}" + {{- end }} + {{- if .Values.sidecar.dashboards.script }} + - name: SCRIPT + value: "{{ .Values.sidecar.dashboards.script }}" + {{- end }} + {{- if .Values.sidecar.dashboards.watchServerTimeout }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.dashboards.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.dashboards.watchClientTimeout }} + - name: WATCH_CLIENT_TIMEOUT + value: "{{ .Values.sidecar.dashboards.watchClientTimeout }}" + {{- end }} +{{- if .Values.sidecar.livenessProbe }} + livenessProbe: +{{ toYaml .Values.livenessProbe | indent 6 }} +{{- end }} +{{- if .Values.sidecar.readinessProbe }} + readinessProbe: +{{ toYaml .Values.readinessProbe | indent 6 }} +{{- end }} + resources: +{{ toYaml .Values.sidecar.resources | indent 6 }} +{{- if .Values.sidecar.securityContext }} + securityContext: +{{- toYaml .Values.sidecar.securityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: sc-dashboard-volume + mountPath: {{ .Values.sidecar.dashboards.folder | quote }} + {{- if .Values.sidecar.dashboards.extraMounts }} + {{- toYaml .Values.sidecar.dashboards.extraMounts | trim | nindent 6}} + {{- end }} +{{- end}} +{{- if .Values.sidecar.datasources.enabled }} + - name: {{ template "grafana.name" . }}-sc-datasources + {{- if .Values.sidecar.image.sha }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + - name: METHOD + value: {{ .Values.sidecar.datasources.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.datasources.label }}" + {{- if .Values.sidecar.datasources.labelValue }} + - name: LABEL_VALUE + value: {{ quote .Values.sidecar.datasources.labelValue }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/datasources" + - name: RESOURCE + value: {{ quote .Values.sidecar.datasources.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- if .Values.sidecar.datasources.searchNamespace }} + - name: NAMESPACE + value: "{{ .Values.sidecar.datasources.searchNamespace | join "," }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.datasources.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.datasources.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} +{{- if .Values.sidecar.livenessProbe }} + livenessProbe: +{{ toYaml .Values.livenessProbe | indent 6 }} +{{- end }} +{{- if .Values.sidecar.readinessProbe }} + readinessProbe: +{{ toYaml .Values.readinessProbe | indent 6 }} +{{- end }} + resources: +{{ toYaml .Values.sidecar.resources | indent 6 }} +{{- if .Values.sidecar.securityContext }} + securityContext: +{{- toYaml .Values.sidecar.securityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" +{{- end}} +{{- if .Values.sidecar.plugins.enabled }} + - name: {{ template "grafana.name" . }}-sc-plugins + {{- if .Values.sidecar.image.sha }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ .Values.sidecar.image.registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + - name: METHOD + value: {{ .Values.sidecar.plugins.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.plugins.label }}" + {{- if .Values.sidecar.plugins.labelValue }} + - name: LABEL_VALUE + value: {{ quote .Values.sidecar.plugins.labelValue }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/plugins" + - name: RESOURCE + value: {{ quote .Values.sidecar.plugins.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- if .Values.sidecar.plugins.searchNamespace }} + - name: NAMESPACE + value: "{{ .Values.sidecar.plugins.searchNamespace | join "," }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.plugins.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.plugins.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} +{{- if .Values.sidecar.livenessProbe }} + livenessProbe: +{{ toYaml .Values.livenessProbe | indent 6 }} +{{- end }} +{{- if .Values.sidecar.readinessProbe }} + readinessProbe: +{{ toYaml .Values.readinessProbe | indent 6 }} +{{- end }} + resources: +{{ toYaml .Values.sidecar.resources | indent 6 }} +{{- if .Values.sidecar.securityContext }} + securityContext: +{{- toYaml .Values.sidecar.securityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: sc-plugins-volume + mountPath: "/etc/grafana/provisioning/plugins" +{{- end}} + - name: {{ .Chart.Name }} + {{- if .Values.image.sha }} + image: "{{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}@sha256:{{ .Values.image.sha }}" + {{- else }} + image: "{{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + {{- if .Values.command }} + command: + {{- range .Values.command }} + - {{ . }} + {{- end }} + {{- end}} +{{- if .Values.containerSecurityContext }} + securityContext: +{{- toYaml .Values.containerSecurityContext | nindent 6 }} +{{- end }} + volumeMounts: + - name: config + mountPath: "/etc/grafana/grafana.ini" + subPath: grafana.ini + {{- if .Values.ldap.enabled }} + - name: ldap + mountPath: "/etc/grafana/ldap.toml" + subPath: ldap.toml + {{- end }} + {{- $root := . }} + {{- range .Values.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + mountPath: {{ tpl .mountPath $root }} + subPath: {{ (tpl .subPath $root) | default "" }} + readOnly: {{ .readOnly }} + {{- end }} + - name: storage + mountPath: "/var/lib/grafana" +{{- if .Values.persistence.subPath }} + subPath: {{ tpl .Values.persistence.subPath . }} +{{- end }} +{{- if .Values.dashboards }} +{{- range $provider, $dashboards := .Values.dashboards }} +{{- range $key, $value := $dashboards }} +{{- if (or (hasKey $value "json") (hasKey $value "file")) }} + - name: dashboards-{{ $provider }} + mountPath: "/var/lib/grafana/dashboards/{{ $provider }}/{{ $key }}.json" + subPath: "{{ $key }}.json" +{{- end }} +{{- end }} +{{- end }} +{{- end -}} +{{- if .Values.dashboardsConfigMaps }} +{{- range (keys .Values.dashboardsConfigMaps | sortAlpha) }} + - name: dashboards-{{ . }} + mountPath: "/var/lib/grafana/dashboards/{{ . }}" +{{- end }} +{{- end }} +{{- if .Values.datasources }} +{{- range (keys .Values.datasources | sortAlpha) }} + - name: config + mountPath: "/etc/grafana/provisioning/datasources/{{ . }}" + subPath: {{ . | quote }} +{{- end }} +{{- end }} +{{- if .Values.notifiers }} +{{- range (keys .Values.notifiers | sortAlpha) }} + - name: config + mountPath: "/etc/grafana/provisioning/notifiers/{{ . }}" + subPath: {{ . | quote }} +{{- end }} +{{- end }} +{{- if .Values.dashboardProviders }} +{{- range (keys .Values.dashboardProviders | sortAlpha) }} + - name: config + mountPath: "/etc/grafana/provisioning/dashboards/{{ . }}" + subPath: {{ . | quote }} +{{- end }} +{{- end }} +{{- if .Values.sidecar.dashboards.enabled }} + - name: sc-dashboard-volume + mountPath: {{ .Values.sidecar.dashboards.folder | quote }} +{{ if .Values.sidecar.dashboards.SCProvider }} + - name: sc-dashboard-provider + mountPath: "/etc/grafana/provisioning/dashboards/sc-dashboardproviders.yaml" + subPath: provider.yaml +{{- end}} +{{- end}} +{{- if .Values.sidecar.datasources.enabled }} + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" +{{- end}} +{{- if .Values.sidecar.plugins.enabled }} + - name: sc-plugins-volume + mountPath: "/etc/grafana/provisioning/plugins" +{{- end}} +{{- if .Values.sidecar.notifiers.enabled }} + - name: sc-notifiers-volume + mountPath: "/etc/grafana/provisioning/notifiers" +{{- end}} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + subPath: {{ .subPath | default "" }} + {{- end }} + {{- range .Values.extraVolumeMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath | default "" }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.extraEmptyDirMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + {{- end }} + ports: + - name: {{ .Values.service.portName }} + containerPort: {{ .Values.service.port }} + protocol: TCP + - name: {{ .Values.podPortName }} + containerPort: 3000 + protocol: TCP + env: + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: GF_SECURITY_ADMIN_USER + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: GF_SECURITY_ADMIN_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if .Values.plugins }} + - name: GF_INSTALL_PLUGINS + valueFrom: + configMapKeyRef: + name: {{ template "grafana.fullname" . }} + key: plugins + {{- end }} + {{- if .Values.smtp.existingSecret }} + - name: GF_SMTP_USER + valueFrom: + secretKeyRef: + name: {{ .Values.smtp.existingSecret }} + key: {{ .Values.smtp.userKey | default "user" }} + - name: GF_SMTP_PASSWORD + valueFrom: + secretKeyRef: + name: {{ .Values.smtp.existingSecret }} + key: {{ .Values.smtp.passwordKey | default "password" }} + {{- end }} + {{- if .Values.imageRenderer.enabled }} + - name: GF_RENDERING_SERVER_URL + value: http://{{ template "grafana.fullname" . }}-image-renderer.{{ template "grafana.namespace" . }}:{{ .Values.imageRenderer.service.port }}/render + - name: GF_RENDERING_CALLBACK_URL + value: {{ .Values.imageRenderer.grafanaProtocol }}://{{ template "grafana.fullname" . }}.{{ template "grafana.namespace" . }}:{{ .Values.service.port }}/{{ .Values.imageRenderer.grafanaSubPath }} + {{- end }} + - name: GF_PATHS_DATA + value: {{ (get .Values "grafana.ini").paths.data }} + - name: GF_PATHS_LOGS + value: {{ (get .Values "grafana.ini").paths.logs }} + - name: GF_PATHS_PLUGINS + value: {{ (get .Values "grafana.ini").paths.plugins }} + - name: GF_PATHS_PROVISIONING + value: {{ (get .Values "grafana.ini").paths.provisioning }} + {{- range $key, $value := .Values.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: +{{ tpl (toYaml $value) $ | indent 10 }} + {{- end }} +{{- range $key, $value := .Values.env }} + - name: "{{ tpl $key $ }}" + value: "{{ tpl (print $value) $ }}" +{{- end }} + {{- if or .Values.envFromSecret (or .Values.envRenderSecret .Values.envFromSecrets) .Values.envFromConfigMaps }} + envFrom: + {{- if .Values.envFromSecret }} + - secretRef: + name: {{ tpl .Values.envFromSecret . }} + {{- end }} + {{- if .Values.envRenderSecret }} + - secretRef: + name: {{ template "grafana.fullname" . }}-env + {{- end }} + {{- range .Values.envFromSecrets }} + - secretRef: + name: {{ tpl .name $ }} + optional: {{ .optional | default false }} + {{- end }} + {{- range .Values.envFromConfigMaps }} + - configMapRef: + name: {{ tpl .name $ }} + optional: {{ .optional | default false }} + {{- end }} + {{- end }} + livenessProbe: +{{ toYaml .Values.livenessProbe | indent 6 }} + readinessProbe: +{{ toYaml .Values.readinessProbe | indent 6 }} +{{- if .Values.lifecycleHooks }} + lifecycle: {{ tpl (.Values.lifecycleHooks | toYaml) . | nindent 6 }} +{{- end }} + resources: +{{ toYaml .Values.resources | indent 6 }} +{{- with .Values.extraContainers }} +{{ tpl . $ | indent 2 }} +{{- end }} +{{- with .Values.nodeSelector }} +nodeSelector: +{{ toYaml . | indent 2 }} +{{- end }} +{{- $root := . }} +{{- with .Values.affinity }} +affinity: +{{ tpl (toYaml .) $root | indent 2 }} +{{- end }} +{{- with .Values.tolerations }} +tolerations: +{{ toYaml . | indent 2 }} +{{- end }} +volumes: + - name: config + configMap: + name: {{ template "grafana.fullname" . }} +{{- $root := . }} +{{- range .Values.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + configMap: + name: {{ tpl .configMap $root }} +{{- end }} + {{- if .Values.dashboards }} + {{- range (keys .Values.dashboards | sortAlpha) }} + - name: dashboards-{{ . }} + configMap: + name: {{ template "grafana.fullname" $ }}-dashboards-{{ . }} + {{- end }} + {{- end }} + {{- if .Values.dashboardsConfigMaps }} + {{ $root := . }} + {{- range $provider, $name := .Values.dashboardsConfigMaps }} + - name: dashboards-{{ $provider }} + configMap: + name: {{ tpl $name $root }} + {{- end }} + {{- end }} + {{- if .Values.ldap.enabled }} + - name: ldap + secret: + {{- if .Values.ldap.existingSecret }} + secretName: {{ .Values.ldap.existingSecret }} + {{- else }} + secretName: {{ template "grafana.fullname" . }} + {{- end }} + items: + - key: ldap-toml + path: ldap.toml + {{- end }} +{{- if and .Values.persistence.enabled (eq .Values.persistence.type "pvc") }} + - name: storage + persistentVolumeClaim: + claimName: {{ tpl (.Values.persistence.existingClaim | default (include "grafana.fullname" .)) . }} +{{- else if and .Values.persistence.enabled (eq .Values.persistence.type "statefulset") }} +# nothing +{{- else }} + - name: storage +{{- if .Values.persistence.inMemory.enabled }} + emptyDir: + medium: Memory +{{- if .Values.persistence.inMemory.sizeLimit }} + sizeLimit: {{ .Values.persistence.inMemory.sizeLimit }} +{{- end -}} +{{- else }} + emptyDir: {} +{{- end -}} +{{- end -}} +{{- if .Values.sidecar.dashboards.enabled }} + - name: sc-dashboard-volume +{{- if .Values.sidecar.dashboards.sizeLimit }} + emptyDir: + sizeLimit: {{ .Values.sidecar.dashboards.sizeLimit }} +{{- else }} + emptyDir: {} +{{- end -}} +{{- if .Values.sidecar.dashboards.SCProvider }} + - name: sc-dashboard-provider + configMap: + name: {{ template "grafana.fullname" . }}-config-dashboards +{{- end }} +{{- end }} +{{- if .Values.sidecar.datasources.enabled }} + - name: sc-datasources-volume +{{- if .Values.sidecar.datasources.sizeLimit }} + emptyDir: + sizeLimit: {{ .Values.sidecar.datasources.sizeLimit }} +{{- else }} + emptyDir: {} +{{- end -}} +{{- end -}} +{{- if .Values.sidecar.plugins.enabled }} + - name: sc-plugins-volume +{{- if .Values.sidecar.plugins.sizeLimit }} + emptyDir: + sizeLimit: {{ .Values.sidecar.plugins.sizeLimit }} +{{- else }} + emptyDir: {} +{{- end -}} +{{- end -}} +{{- if .Values.sidecar.notifiers.enabled }} + - name: sc-notifiers-volume +{{- if .Values.sidecar.notifiers.sizeLimit }} + emptyDir: + sizeLimit: {{ .Values.sidecar.notifiers.sizeLimit }} +{{- else }} + emptyDir: {} +{{- end -}} +{{- end -}} +{{- range .Values.extraSecretMounts }} +{{- if .secretName }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + defaultMode: {{ .defaultMode }} +{{- else if .projected }} + - name: {{ .name }} + projected: {{- toYaml .projected | nindent 6 }} +{{- else if .csi }} + - name: {{ .name }} + csi: {{- toYaml .csi | nindent 6 }} +{{- end }} +{{- end }} +{{- range .Values.extraVolumeMounts }} + - name: {{ .name }} + {{- if .existingClaim }} + persistentVolumeClaim: + claimName: {{ .existingClaim }} + {{- else if .hostPath }} + hostPath: + path: {{ .hostPath }} + {{- else }} + emptyDir: {} + {{- end }} +{{- end }} +{{- range .Values.extraEmptyDirMounts }} + - name: {{ .name }} + emptyDir: {} +{{- end -}} +{{- if .Values.extraContainerVolumes }} +{{ toYaml .Values.extraContainerVolumes | indent 2 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrole.yaml new file mode 100644 index 000000000..f09e06563 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrole.yaml @@ -0,0 +1,25 @@ +{{- if and .Values.rbac.create (not .Values.rbac.namespaced) (not .Values.rbac.useExistingRole) }} +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} + name: {{ template "grafana.fullname" . }}-clusterrole +{{- if or .Values.sidecar.dashboards.enabled (or .Values.sidecar.datasources.enabled .Values.rbac.extraClusterRoleRules) }} +rules: +{{- if or .Values.sidecar.dashboards.enabled .Values.sidecar.datasources.enabled }} +- apiGroups: [""] # "" indicates the core API group + resources: ["configmaps", "secrets"] + verbs: ["get", "watch", "list"] +{{- end}} +{{- with .Values.rbac.extraClusterRoleRules }} +{{ toYaml . | indent 0 }} +{{- end}} +{{- else }} +rules: [] +{{- end}} +{{- end}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..4accbfac0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/clusterrolebinding.yaml @@ -0,0 +1,24 @@ +{{- if and .Values.rbac.create (not .Values.rbac.namespaced) }} +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ template "grafana.fullname" . }}-clusterrolebinding + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +subjects: + - kind: ServiceAccount + name: {{ template "grafana.serviceAccountName" . }} + namespace: {{ template "grafana.namespace" . }} +roleRef: + kind: ClusterRole +{{- if (not .Values.rbac.useExistingRole) }} + name: {{ template "grafana.fullname" . }}-clusterrole +{{- else }} + name: {{ .Values.rbac.useExistingRole }} +{{- end }} + apiGroup: rbac.authorization.k8s.io +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap-dashboard-provider.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap-dashboard-provider.yaml new file mode 100644 index 000000000..65d73858e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap-dashboard-provider.yaml @@ -0,0 +1,29 @@ +{{- if .Values.sidecar.dashboards.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} + name: {{ template "grafana.fullname" . }}-config-dashboards + namespace: {{ template "grafana.namespace" . }} +data: + provider.yaml: |- + apiVersion: 1 + providers: + - name: '{{ .Values.sidecar.dashboards.provider.name }}' + orgId: {{ .Values.sidecar.dashboards.provider.orgid }} + {{- if not .Values.sidecar.dashboards.provider.foldersFromFilesStructure }} + folder: '{{ .Values.sidecar.dashboards.provider.folder }}' + {{- end}} + type: {{ .Values.sidecar.dashboards.provider.type }} + disableDeletion: {{ .Values.sidecar.dashboards.provider.disableDelete }} + allowUiUpdates: {{ .Values.sidecar.dashboards.provider.allowUiUpdates }} + updateIntervalSeconds: {{ .Values.sidecar.dashboards.provider.updateIntervalSeconds | default 30 }} + options: + foldersFromFilesStructure: {{ .Values.sidecar.dashboards.provider.foldersFromFilesStructure }} + path: {{ .Values.sidecar.dashboards.folder }}{{- with .Values.sidecar.dashboards.defaultFolderName }}/{{ . }}{{- end }} +{{- end}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap.yaml new file mode 100644 index 000000000..401e2aaa6 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/configmap.yaml @@ -0,0 +1,88 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +data: +{{- if .Values.plugins }} + plugins: {{ join "," .Values.plugins }} +{{- end }} + grafana.ini: | +{{- range $key, $value := index .Values "grafana.ini" }} + [{{ $key }}] + {{- range $elem, $elemVal := $value }} + {{- if kindIs "invalid" $elemVal }} + {{ $elem }} = + {{- else if kindIs "string" $elemVal }} + {{ $elem }} = {{ tpl $elemVal $ }} + {{- else }} + {{ $elem }} = {{ $elemVal }} + {{- end }} + {{- end }} +{{- end }} + +{{- if .Values.datasources }} +{{ $root := . }} + {{- range $key, $value := .Values.datasources }} + {{ $key }}: | +{{ tpl (toYaml $value | indent 4) $root }} + {{- end -}} +{{- end -}} + +{{- if .Values.notifiers }} + {{- range $key, $value := .Values.notifiers }} + {{ $key }}: | +{{ toYaml $value | indent 4 }} + {{- end -}} +{{- end -}} + +{{- if .Values.dashboardProviders }} + {{- range $key, $value := .Values.dashboardProviders }} + {{ $key }}: | +{{ toYaml $value | indent 4 }} + {{- end -}} +{{- end -}} + +{{- if .Values.dashboards }} + download_dashboards.sh: | + #!/usr/bin/env sh + set -euf + {{- if .Values.dashboardProviders }} + {{- range $key, $value := .Values.dashboardProviders }} + {{- range $value.providers }} + mkdir -p {{ .options.path }} + {{- end }} + {{- end }} + {{- end }} + {{ $dashboardProviders := .Values.dashboardProviders }} + {{- range $provider, $dashboards := .Values.dashboards }} + {{- range $key, $value := $dashboards }} + {{- if (or (hasKey $value "gnetId") (hasKey $value "url")) }} + curl -skf \ + --connect-timeout 60 \ + --max-time 60 \ + {{- if not $value.b64content }} + -H "Accept: application/json" \ + {{- if $value.token }} + -H "Authorization: token {{ $value.token }}" \ + {{- end }} + -H "Content-Type: application/json;charset=UTF-8" \ + {{ end }} + {{- $dpPath := "" -}} + {{- range $kd := (index $dashboardProviders "dashboardproviders.yaml").providers -}} + {{- if eq $kd.name $provider -}} + {{- $dpPath = $kd.options.path -}} + {{- end -}} + {{- end -}} + {{- if $value.url -}}"{{ $value.url }}"{{- else -}}"https://grafana.com/api/dashboards/{{ $value.gnetId }}/revisions/{{- if $value.revision -}}{{ $value.revision }}{{- else -}}1{{- end -}}/download"{{- end -}}{{ if $value.datasource }} | sed '/-- .* --/! s/"datasource":.*,/"datasource": "{{ $value.datasource }}",/g'{{ end }}{{- if $value.b64content -}} | base64 -d {{- end -}} \ + > "{{- if $dpPath -}}{{ $dpPath }}{{- else -}}/var/lib/grafana/dashboards/{{ $provider }}{{- end -}}/{{ $key }}.json" + {{- end }} + {{- end -}} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/dashboards-json-configmap.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/dashboards-json-configmap.yaml new file mode 100644 index 000000000..24212b736 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/dashboards-json-configmap.yaml @@ -0,0 +1,36 @@ +{{- if .Values.dashboards }} +{{ $files := .Files }} +{{- range $provider, $dashboards := .Values.dashboards }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "grafana.fullname" $ }}-dashboards-{{ $provider }} + namespace: {{ template "grafana.namespace" $ }} + labels: + {{- include "grafana.labels" $ | nindent 4 }} + dashboard-provider: {{ $provider }} + grafana_dashboard: tvm +{{- if $dashboards }} +data: +{{- $dashboardFound := false }} +{{- range $key, $value := $dashboards }} +{{- if (or (hasKey $value "json") (hasKey $value "file")) }} +{{- $dashboardFound = true }} +{{ print $key | indent 2 }}.json: +{{- if hasKey $value "json" }} + |- +{{ $value.json | indent 6 }} +{{- end }} +{{- if hasKey $value "file" }} +{{ toYaml ( $files.Get $value.file ) | indent 4}} +{{- end }} +{{- end }} +{{- end }} +{{- if not $dashboardFound }} + {} +{{- end }} +{{- end }} +--- +{{- end }} + +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/deployment.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/deployment.yaml new file mode 100644 index 000000000..8dbe5e107 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/deployment.yaml @@ -0,0 +1,50 @@ +{{ if (or (not .Values.persistence.enabled) (eq .Values.persistence.type "pvc")) }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- if .Values.labels }} +{{ toYaml .Values.labels | indent 4 }} +{{- end }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + {{- if and (not .Values.autoscaling.enabled) (.Values.replicas) }} + replicas: {{ .Values.replicas }} + {{- end }} + revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} +{{- with .Values.deploymentStrategy }} + strategy: +{{ toYaml . | trim | indent 4 }} +{{- end }} + template: + metadata: + labels: + {{- include "grafana.selectorLabels" . | nindent 8 }} +{{- with .Values.podLabels }} +{{ toYaml . | indent 8 }} +{{- end }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + checksum/dashboards-json-config: {{ include (print $.Template.BasePath "/dashboards-json-configmap.yaml") . | sha256sum }} + checksum/sc-dashboard-provider-config: {{ include (print $.Template.BasePath "/configmap-dashboard-provider.yaml") . | sha256sum }} +{{- if and (or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret))) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} +{{- end }} +{{- if .Values.envRenderSecret }} + checksum/secret-env: {{ include (print $.Template.BasePath "/secret-env.yaml") . | sha256sum }} +{{- end }} +{{- with .Values.podAnnotations }} +{{ toYaml . | indent 8 }} +{{- end }} + spec: + {{- include "grafana.pod" . | nindent 6 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/extra-manifests.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/extra-manifests.yaml new file mode 100644 index 000000000..a9bb3b6ba --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraObjects }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/headless-service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/headless-service.yaml new file mode 100644 index 000000000..1df42e967 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/headless-service.yaml @@ -0,0 +1,22 @@ +{{- if or .Values.headlessService (and .Values.persistence.enabled (not .Values.persistence.existingClaim) (eq .Values.persistence.type "statefulset"))}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "grafana.fullname" . }}-headless + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + clusterIP: None + selector: + {{- include "grafana.selectorLabels" . | nindent 4 }} + type: ClusterIP + ports: + - protocol: TCP + port: 3000 + targetPort: 3000 +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/hpa.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/hpa.yaml new file mode 100644 index 000000000..9c186d74a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/hpa.yaml @@ -0,0 +1,20 @@ +{{- if .Values.autoscaling.enabled }} +apiVersion: autoscaling/v2beta1 +kind: HorizontalPodAutoscaler +metadata: + name: {{ template "grafana.fullname" . }} + labels: + app.kubernetes.io/name: {{ template "grafana.name" . }} + helm.sh/chart: {{ template "grafana.chart" . }} + app.kubernetes.io/managed-by: {{ .Release.Service }} + app.kubernetes.io/instance: {{ .Release.Name }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ template "grafana.fullname" . }} + minReplicas: {{ .Values.autoscaling.minReplicas }} + maxReplicas: {{ .Values.autoscaling.maxReplicas }} + metrics: +{{ toYaml .Values.autoscaling.metrics | indent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-deployment.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-deployment.yaml new file mode 100644 index 000000000..acf262e9d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-deployment.yaml @@ -0,0 +1,121 @@ +{{ if .Values.imageRenderer.enabled }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "grafana.fullname" . }}-image-renderer + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.imageRenderer.labels" . | nindent 4 }} +{{- if .Values.imageRenderer.labels }} +{{ toYaml .Values.imageRenderer.labels | indent 4 }} +{{- end }} +{{- with .Values.imageRenderer.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + replicas: {{ .Values.imageRenderer.replicas }} + revisionHistoryLimit: {{ .Values.imageRenderer.revisionHistoryLimit }} + selector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} +{{- with .Values.imageRenderer.deploymentStrategy }} + strategy: +{{ toYaml . | trim | indent 4 }} +{{- end }} + template: + metadata: + labels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 8 }} +{{- with .Values.imageRenderer.podLabels }} +{{ toYaml . | indent 8 }} +{{- end }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} +{{- with .Values.imageRenderer.podAnnotations }} +{{ toYaml . | indent 8 }} +{{- end }} + spec: + + {{- if .Values.imageRenderer.schedulerName }} + schedulerName: "{{ .Values.imageRenderer.schedulerName }}" + {{- end }} + {{- if .Values.imageRenderer.serviceAccountName }} + serviceAccountName: "{{ .Values.imageRenderer.serviceAccountName }}" + {{- end }} + {{- if .Values.imageRenderer.securityContext }} + securityContext: + {{- toYaml .Values.imageRenderer.securityContext | nindent 8 }} + {{- end }} + {{- if .Values.imageRenderer.hostAliases }} + hostAliases: + {{- toYaml .Values.imageRenderer.hostAliases | nindent 8 }} + {{- end }} + {{- if .Values.imageRenderer.priorityClassName }} + priorityClassName: {{ .Values.imageRenderer.priorityClassName }} + {{- end }} + {{- if .Values.imageRenderer.image.pullSecrets }} + imagePullSecrets: + {{- $root := . }} + {{- range .Values.imageRenderer.image.pullSecrets }} + - name: {{ tpl . $root }} + {{- end}} + {{- end }} + containers: + - name: {{ .Chart.Name }}-image-renderer + {{- if .Values.imageRenderer.image.sha }} + image: "{{ .Values.imageRenderer.image.registry }}/{{ .Values.imageRenderer.image.repository }}:{{ .Values.imageRenderer.image.tag }}@sha256:{{ .Values.imageRenderer.image.sha }}" + {{- else }} + image: "{{ .Values.imageRenderer.image.registry }}/{{ .Values.imageRenderer.image.repository }}:{{ .Values.imageRenderer.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.imageRenderer.image.pullPolicy }} + {{- if .Values.imageRenderer.command }} + command: + {{- range .Values.imageRenderer.command }} + - {{ . }} + {{- end }} + {{- end}} + ports: + - name: {{ .Values.imageRenderer.service.portName }} + containerPort: {{ .Values.imageRenderer.service.port }} + protocol: TCP + livenessProbe: + httpGet: + path: / + port: {{ .Values.imageRenderer.service.portName }} + env: + - name: HTTP_PORT + value: {{ .Values.imageRenderer.service.port | quote }} + {{- range $key, $value := .Values.imageRenderer.env }} + - name: {{ $key | quote }} + value: {{ $value | quote }} + {{- end }} + securityContext: + capabilities: + drop: ['all'] + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + volumeMounts: + - mountPath: /tmp + name: image-renderer-tmpfs + {{- with .Values.imageRenderer.resources }} + resources: +{{ toYaml . | indent 12 }} + {{- end }} + {{- with .Values.imageRenderer.nodeSelector }} + nodeSelector: +{{ toYaml . | indent 8 }} + {{- end }} + {{- $root := . }} + {{- with .Values.imageRenderer.affinity }} + affinity: +{{ tpl (toYaml .) $root | indent 8 }} + {{- end }} + {{- with .Values.imageRenderer.tolerations }} + tolerations: +{{ toYaml . | indent 8 }} + {{- end }} + volumes: + - name: image-renderer-tmpfs + emptyDir: {} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-network-policy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-network-policy.yaml new file mode 100644 index 000000000..f8ca73aab --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-network-policy.yaml @@ -0,0 +1,76 @@ +{{- if and (.Values.imageRenderer.enabled) (.Values.imageRenderer.networkPolicy.limitIngress) }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "grafana.fullname" . }}-image-renderer-ingress + namespace: {{ template "grafana.namespace" . }} + annotations: + comment: Limit image-renderer ingress traffic from grafana +spec: + podSelector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + {{- if .Values.imageRenderer.podLabels }} + {{ toYaml .Values.imageRenderer.podLabels | nindent 6 }} + {{- end }} + + policyTypes: + - Ingress + ingress: + - ports: + - port: {{ .Values.imageRenderer.service.port }} + protocol: TCP + from: + - namespaceSelector: + matchLabels: + name: {{ template "grafana.namespace" . }} + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 14 }} + {{- if .Values.podLabels }} + {{ toYaml .Values.podLabels | nindent 14 }} + {{- end }} +{{ end }} + +{{- if and (.Values.imageRenderer.enabled) (.Values.imageRenderer.networkPolicy.limitEgress) }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "grafana.fullname" . }}-image-renderer-egress + namespace: {{ template "grafana.namespace" . }} + annotations: + comment: Limit image-renderer egress traffic to grafana +spec: + podSelector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + {{- if .Values.imageRenderer.podLabels }} + {{ toYaml .Values.imageRenderer.podLabels | nindent 6 }} + {{- end }} + + policyTypes: + - Egress + egress: + # allow dns resolution + - ports: + - port: 53 + protocol: UDP + - port: 53 + protocol: TCP + # talk only to grafana + - ports: + - port: {{ .Values.service.port }} + protocol: TCP + to: + - namespaceSelector: + matchLabels: + name: {{ template "grafana.namespace" . }} + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 14 }} + {{- if .Values.podLabels }} + {{ toYaml .Values.podLabels | nindent 14 }} + {{- end }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-service.yaml new file mode 100644 index 000000000..f29586c3a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/image-renderer-service.yaml @@ -0,0 +1,30 @@ +{{ if .Values.imageRenderer.enabled }} +{{ if .Values.imageRenderer.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "grafana.fullname" . }}-image-renderer + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.imageRenderer.labels" . | nindent 4 }} +{{- if .Values.imageRenderer.service.labels }} +{{ toYaml .Values.imageRenderer.service.labels | indent 4 }} +{{- end }} +{{- with .Values.imageRenderer.service.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + type: ClusterIP + {{- if .Values.imageRenderer.service.clusterIP }} + clusterIP: {{ .Values.imageRenderer.service.clusterIP }} + {{end}} + ports: + - name: {{ .Values.imageRenderer.service.portName }} + port: {{ .Values.imageRenderer.service.port }} + protocol: TCP + targetPort: {{ .Values.imageRenderer.service.targetPort }} + selector: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 4 }} +{{ end }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/ingress.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/ingress.yaml new file mode 100644 index 000000000..7699cecaa --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/ingress.yaml @@ -0,0 +1,78 @@ +{{- if .Values.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "grafana.ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "grafana.ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "grafana.ingress.supportsPathType" .) "true" -}} +{{- $fullName := include "grafana.fullname" . -}} +{{- $servicePort := .Values.service.port -}} +{{- $ingressPath := .Values.ingress.path -}} +{{- $ingressPathType := .Values.ingress.pathType -}} +{{- $extraPaths := .Values.ingress.extraPaths -}} +apiVersion: {{ include "grafana.ingress.apiVersion" . }} +kind: Ingress +metadata: + name: {{ $fullName }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- if .Values.ingress.labels }} +{{ toYaml .Values.ingress.labels | indent 4 }} +{{- end }} + {{- if .Values.ingress.annotations }} + annotations: + {{- range $key, $value := .Values.ingress.annotations }} + {{ $key }}: {{ tpl $value $ | quote }} + {{- end }} + {{- end }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.ingress.ingressClassName }} + ingressClassName: {{ .Values.ingress.ingressClassName }} + {{- end -}} +{{- if .Values.ingress.tls }} + tls: +{{ tpl (toYaml .Values.ingress.tls) $ | indent 4 }} +{{- end }} + rules: + {{- if .Values.ingress.hosts }} + {{- range .Values.ingress.hosts }} + - host: {{ tpl . $}} + http: + paths: +{{- if $extraPaths }} +{{ toYaml $extraPaths | indent 10 }} +{{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $fullName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $fullName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end }} + {{- else }} + - http: + paths: + - backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $fullName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $fullName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- if $ingressPath }} + path: {{ $ingressPath }} + {{- end }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + {{- end -}} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/networkpolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/networkpolicy.yaml new file mode 100644 index 000000000..fc243828e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/networkpolicy.yaml @@ -0,0 +1,37 @@ +{{- if .Values.networkPolicy.enabled }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- if .Values.labels }} +{{ toYaml .Values.labels | indent 4 }} +{{- end }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + ingress: + - ports: + - port: {{ .Values.service.targetPort }} + {{- if not .Values.networkPolicy.allowExternal }} + from: + - podSelector: + matchLabels: + {{ template "grafana.fullname" . }}-client: "true" + {{- if .Values.networkPolicy.explicitNamespacesSelector }} + namespaceSelector: + {{ toYaml .Values.networkPolicy.explicitNamespacesSelector | indent 12 }} + {{- end }} + - podSelector: + matchLabels: + {{- include "grafana.labels" . | nindent 14 }} + role: read + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/poddisruptionbudget.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/poddisruptionbudget.yaml new file mode 100644 index 000000000..61813a436 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/poddisruptionbudget.yaml @@ -0,0 +1,22 @@ +{{- if .Values.podDisruptionBudget }} +apiVersion: policy/v1beta1 +kind: PodDisruptionBudget +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- if .Values.labels }} +{{ toYaml .Values.labels | indent 4 }} +{{- end }} +spec: +{{- if .Values.podDisruptionBudget.minAvailable }} + minAvailable: {{ .Values.podDisruptionBudget.minAvailable }} +{{- end }} +{{- if .Values.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ .Values.podDisruptionBudget.maxUnavailable }} +{{- end }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/podsecuritypolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..7de6c021d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/podsecuritypolicy.yaml @@ -0,0 +1,49 @@ +{{- if .Values.rbac.pspEnabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ template "grafana.fullname" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + annotations: + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default,runtime/default' + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + {{- if .Values.rbac.pspUseAppArmor }} + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + {{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + # Default set from Docker, with DAC_OVERRIDE and CHOWN + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'csi' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/pvc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/pvc.yaml new file mode 100644 index 000000000..8d93f5c23 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/pvc.yaml @@ -0,0 +1,33 @@ +{{- if and .Values.persistence.enabled (not .Values.persistence.existingClaim) (eq .Values.persistence.type "pvc")}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.persistence.annotations }} + annotations: +{{ toYaml . | indent 4 }} + {{- end }} + {{- with .Values.persistence.finalizers }} + finalizers: +{{ toYaml . | indent 4 }} + {{- end }} +spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{- if .Values.persistence.storageClassName }} + storageClassName: {{ .Values.persistence.storageClassName }} + {{- end -}} + {{- with .Values.persistence.selectorLabels }} + selector: + matchLabels: +{{ toYaml . | indent 6 }} + {{- end }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/role.yaml new file mode 100644 index 000000000..6a1890fb9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/role.yaml @@ -0,0 +1,32 @@ +{{- if and .Values.rbac.create (not .Values.rbac.useExistingRole) -}} +apiVersion: {{ template "grafana.rbac.apiVersion" . }} +kind: Role +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +{{- if or .Values.rbac.pspEnabled (and .Values.rbac.namespaced (or .Values.sidecar.dashboards.enabled (or .Values.sidecar.datasources.enabled .Values.rbac.extraRoleRules))) }} +rules: +{{- if .Values.rbac.pspEnabled }} +- apiGroups: ['extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: [{{ template "grafana.fullname" . }}] +{{- end }} +{{- if and .Values.rbac.namespaced (or .Values.sidecar.dashboards.enabled .Values.sidecar.datasources.enabled) }} +- apiGroups: [""] # "" indicates the core API group + resources: ["configmaps", "secrets"] + verbs: ["get", "watch", "list"] +{{- end }} +{{- with .Values.rbac.extraRoleRules }} +{{ toYaml . | indent 0 }} +{{- end}} +{{- else }} +rules: [] +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/rolebinding.yaml new file mode 100644 index 000000000..e0107255e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/rolebinding.yaml @@ -0,0 +1,25 @@ +{{- if .Values.rbac.create -}} +apiVersion: {{ template "grafana.rbac.apiVersion" . }} +kind: RoleBinding +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role +{{- if (not .Values.rbac.useExistingRole) }} + name: {{ template "grafana.fullname" . }} +{{- else }} + name: {{ .Values.rbac.useExistingRole }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "grafana.serviceAccountName" . }} + namespace: {{ template "grafana.namespace" . }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret-env.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret-env.yaml new file mode 100644 index 000000000..5c09313e6 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret-env.yaml @@ -0,0 +1,14 @@ +{{- if .Values.envRenderSecret }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "grafana.fullname" . }}-env + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +type: Opaque +data: +{{- range $key, $val := .Values.envRenderSecret }} + {{ $key }}: {{ $val | b64enc | quote }} +{{- end -}} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret.yaml new file mode 100644 index 000000000..c8aa750ac --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/secret.yaml @@ -0,0 +1,26 @@ +{{- if or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret)) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +type: Opaque +data: + {{- if and (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) }} + admin-user: {{ .Values.adminUser | b64enc | quote }} + {{- if .Values.adminPassword }} + admin-password: {{ .Values.adminPassword | b64enc | quote }} + {{- else }} + admin-password: {{ template "grafana.password" . }} + {{- end }} + {{- end }} + {{- if not .Values.ldap.existingSecret }} + ldap-toml: {{ tpl .Values.ldap.config $ | b64enc | quote }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/service.yaml new file mode 100644 index 000000000..ba84ef970 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/service.yaml @@ -0,0 +1,51 @@ +{{ if .Values.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- if .Values.service.labels }} +{{ toYaml .Values.service.labels | indent 4 }} +{{- end }} +{{- with .Values.service.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: +{{- if (or (eq .Values.service.type "ClusterIP") (empty .Values.service.type)) }} + type: ClusterIP + {{- if .Values.service.clusterIP }} + clusterIP: {{ .Values.service.clusterIP }} + {{end}} +{{- else if eq .Values.service.type "LoadBalancer" }} + type: {{ .Values.service.type }} + {{- if .Values.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.service.loadBalancerIP }} + {{- end }} + {{- if .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ toYaml .Values.service.loadBalancerSourceRanges | indent 4 }} + {{- end -}} +{{- else }} + type: {{ .Values.service.type }} +{{- end }} +{{- if .Values.service.externalIPs }} + externalIPs: +{{ toYaml .Values.service.externalIPs | indent 4 }} +{{- end }} + ports: + - name: {{ .Values.service.portName }} + port: {{ .Values.service.port }} + protocol: TCP + targetPort: {{ .Values.service.targetPort }} +{{ if (and (eq .Values.service.type "NodePort") (not (empty .Values.service.nodePort))) }} + nodePort: {{.Values.service.nodePort}} +{{ end }} + {{- if .Values.extraExposePorts }} + {{- tpl (toYaml .Values.extraExposePorts) . | indent 4 }} + {{- end }} + selector: + {{- include "grafana.selectorLabels" . | nindent 4 }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/serviceaccount.yaml new file mode 100644 index 000000000..4ccee15ed --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/serviceaccount.yaml @@ -0,0 +1,14 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- $root := . }} +{{- with .Values.serviceAccount.annotations }} + annotations: +{{ tpl (toYaml . | indent 4) $root }} +{{- end }} + name: {{ template "grafana.serviceAccountName" . }} + namespace: {{ template "grafana.namespace" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/servicemonitor.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/servicemonitor.yaml new file mode 100644 index 000000000..a18c6d336 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/servicemonitor.yaml @@ -0,0 +1,44 @@ +{{- if .Values.serviceMonitor.enabled }} +--- +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ template "grafana.fullname" . }} + {{- if .Values.serviceMonitor.namespace }} + namespace: {{ .Values.serviceMonitor.namespace }} + {{- else }} + namespace: {{ template "grafana.namespace" . }} + {{- end }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- if .Values.serviceMonitor.labels }} + {{- toYaml .Values.serviceMonitor.labels | nindent 4 }} + {{- end }} +spec: + endpoints: + - port: {{ .Values.service.portName }} + {{- with .Values.serviceMonitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + honorLabels: true + path: {{ .Values.serviceMonitor.path }} + scheme: {{ .Values.serviceMonitor.scheme }} + {{- if .Values.serviceMonitor.tlsConfig }} + tlsConfig: + {{- toYaml .Values.serviceMonitor.tlsConfig | nindent 6 }} + {{- end }} + {{- if .Values.serviceMonitor.relabelings }} + relabelings: + {{- toYaml .Values.serviceMonitor.relabelings | nindent 4 }} + {{- end }} + jobLabel: "{{ .Release.Name }}" + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 8 }} + namespaceSelector: + matchNames: + - {{ .Release.Namespace }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/statefulset.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/statefulset.yaml new file mode 100644 index 000000000..ad3dd0696 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/statefulset.yaml @@ -0,0 +1,52 @@ +{{- if and .Values.persistence.enabled (not .Values.persistence.existingClaim) (eq .Values.persistence.type "statefulset")}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: + name: {{ template "grafana.fullname" . }} + namespace: {{ template "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +{{- with .Values.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + replicas: {{ .Values.replicas }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + serviceName: {{ template "grafana.fullname" . }}-headless + template: + metadata: + labels: + {{- include "grafana.selectorLabels" . | nindent 8 }} +{{- with .Values.podLabels }} +{{ toYaml . | indent 8 }} +{{- end }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + checksum/dashboards-json-config: {{ include (print $.Template.BasePath "/dashboards-json-configmap.yaml") . | sha256sum }} + checksum/sc-dashboard-provider-config: {{ include (print $.Template.BasePath "/configmap-dashboard-provider.yaml") . | sha256sum }} + {{- if and (or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret))) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} +{{- end }} +{{- with .Values.podAnnotations }} +{{ toYaml . | indent 8 }} +{{- end }} + spec: + {{- include "grafana.pod" . | nindent 6 }} + volumeClaimTemplates: + - metadata: + name: storage + spec: + accessModes: {{ .Values.persistence.accessModes }} + storageClassName: {{ .Values.persistence.storageClassName }} + resources: + requests: + storage: {{ .Values.persistence.size }} + {{- with .Values.persistence.selectorLabels }} + selector: + matchLabels: +{{ toYaml . | indent 10 }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/user-secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/user-secret.yaml new file mode 100644 index 000000000..3e9703fff --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/templates/user-secret.yaml @@ -0,0 +1,9 @@ +apiVersion: v1 +kind: Secret +metadata: + name: k8s-triliovault-operator-grafana + namespace: {{ template "grafana.namespace" . }} +type: Opaque +data: + admin-user: YWRtaW4= + admin-password: YWRtaW4xMjM= diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/values.yaml new file mode 100644 index 000000000..5d50d4443 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/values.yaml @@ -0,0 +1,938 @@ +rbac: + create: true + ## Use an existing ClusterRole/Role (depending on rbac.namespaced false/true) + # useExistingRole: name-of-some-(cluster)role + pspEnabled: false + pspUseAppArmor: false + namespaced: false + extraRoleRules: [] + # - apiGroups: [] + # resources: [] + # verbs: [] + extraClusterRoleRules: [] + # - apiGroups: [] + # resources: [] + # verbs: [] +serviceAccount: + create: true + name: + nameTest: +## Service account annotations. Can be templated. +# annotations: +# eks.amazonaws.com/role-arn: arn:aws:iam::123456789000:role/iam-role-name-here + autoMount: true + +replicas: 1 + +## Create a headless service for the deployment +headlessService: false + +## Create HorizontalPodAutoscaler object for deployment type +# +autoscaling: + enabled: false +# minReplicas: 1 +# maxReplicas: 10 +# metrics: +# - type: Resource +# resource: +# name: cpu +# targetAverageUtilization: 60 +# - type: Resource +# resource: +# name: memory +# targetAverageUtilization: 60 + +## See `kubectl explain poddisruptionbudget.spec` for more +## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ +podDisruptionBudget: {} +# minAvailable: 1 +# maxUnavailable: 1 + +## See `kubectl explain deployment.spec.strategy` for more +## ref: https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy +deploymentStrategy: + type: RollingUpdate + +readinessProbe: + httpGet: + path: /api/health + port: 3000 + +livenessProbe: + httpGet: + path: /api/health + port: 3000 + initialDelaySeconds: 60 + timeoutSeconds: 30 + failureThreshold: 10 + +## Use an alternate scheduler, e.g. "stork". +## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ +## +# schedulerName: "default-scheduler" + +image: + registry: docker.io + repository: grafana/grafana + tag: 8.5.0 + sha: "" + pullPolicy: IfNotPresent + + ## Optionally specify an array of imagePullSecrets. + ## Secrets must be manually created in the namespace. + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + ## Can be templated. + ## + # pullSecrets: + # - myRegistrKeySecretName + +testFramework: + enabled: true + registry: docker.io + image: "bats/bats" + tag: "v1.4.1" + imagePullPolicy: IfNotPresent + securityContext: {} + +securityContext: + runAsUser: 472 + runAsGroup: 472 + fsGroup: 472 + +containerSecurityContext: + {} + +# Extra configmaps to mount in grafana pods +# Values are templated. +extraConfigmapMounts: [] + # - name: certs-configmap + # mountPath: /etc/grafana/ssl/ + # subPath: certificates.crt # (optional) + # configMap: certs-configmap + # readOnly: true + + +extraEmptyDirMounts: [] + # - name: provisioning-notifiers + # mountPath: /etc/grafana/provisioning/notifiers + + +# Apply extra labels to common labels. +extraLabels: {} + +## Assign a PriorityClassName to pods if set +# priorityClassName: + +downloadDashboardsImage: + registry: docker.io + repository: curlimages/curl + tag: 7.73.0 + sha: "" + pullPolicy: IfNotPresent + +downloadDashboards: + env: {} + envFromSecret: "" + resources: {} + +## Pod Annotations +# podAnnotations: {} + +## Pod Labels +# podLabels: {} + +podPortName: grafana + +## Deployment annotations +annotations: + ignore-check.kube-linter.io/privileged-ports : "This deployment needs to run on privileged ports 80" + ignore-check.kube-linter.io/read-secret-from-env-var : "This deployment needs to read secret from env variable for grafana admin user and password" + +## Expose the grafana service to be accessed from outside the cluster (LoadBalancer service). +## or access it from within the cluster (ClusterIP service). Set the service type and the port to serve it. +## ref: http://kubernetes.io/docs/user-guide/services/ +## +service: + enabled: true + type: ClusterIP + port: 80 + targetPort: 3000 + # targetPort: 4181 To be used with a proxy extraContainer + annotations: {} + labels: {} + portName: service + +serviceMonitor: + ## If true, a ServiceMonitor CRD is created for a prometheus operator + ## https://github.com/coreos/prometheus-operator + ## + enabled: false + path: /metrics + # namespace: monitoring (defaults to use the namespace this chart is deployed to) + labels: {} + interval: 1m + scheme: http + tlsConfig: {} + scrapeTimeout: 30s + relabelings: [] + +extraExposePorts: [] + # - name: keycloak + # port: 8080 + # targetPort: 8080 + # type: ClusterIP + +# overrides pod.spec.hostAliases in the grafana deployment's pods +hostAliases: [] + # - ip: "1.2.3.4" + # hostnames: + # - "my.host.com" + +ingress: + enabled: false + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + # Values can be templated + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + labels: {} + path: / + + # pathType is only for k8s >= 1.1= + pathType: Prefix + + hosts: + - chart-example.local + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + ## Or for k8s > 1.19 + # - path: /* + # pathType: Prefix + # backend: + # service: + # name: ssl-redirect + # port: + # name: use-annotation + + + tls: [] + # - secretName: chart-example-tls + # hosts: + # - chart-example.local + +resources: + limits: + cpu: 1000m + memory: 500Mi + requests: + cpu: 200m + memory: 256Mi + +## Node labels for pod assignment +## ref: https://kubernetes.io/docs/user-guide/node-selection/ +# +nodeSelector: {} + +## Tolerations for pod assignment +## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +## +tolerations: [] + +## Affinity for pod assignment (evaluated as template) +## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity +## +affinity: {} + +## Additional init containers (evaluated as template) +## ref: https://kubernetes.io/docs/concepts/workloads/pods/init-containers/ +## +extraInitContainers: [] + +## Enable an Specify container in extraContainers. This is meant to allow adding an authentication proxy to a grafana pod +extraContainers: "" +# extraContainers: | +# - name: proxy +# image: quay.io/gambol99/keycloak-proxy:latest +# args: +# - -provider=github +# - -client-id= +# - -client-secret= +# - -github-org= +# - -email-domain=* +# - -cookie-secret= +# - -http-address=http://0.0.0.0:4181 +# - -upstream-url=http://127.0.0.1:3000 +# ports: +# - name: proxy-web +# containerPort: 4181 + +## Volumes that can be used in init containers that will not be mounted to deployment pods +extraContainerVolumes: [] +# - name: volume-from-secret +# secret: +# secretName: secret-to-mount +# - name: empty-dir-volume +# emptyDir: {} + +## Enable persistence using Persistent Volume Claims +## ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ +## +persistence: + type: pvc + enabled: false + # storageClassName: default + accessModes: + - ReadWriteOnce + size: 10Gi + # annotations: {} + finalizers: + - kubernetes.io/pvc-protection + # selectorLabels: {} + ## Sub-directory of the PV to mount. Can be templated. + # subPath: "" + ## Name of an existing PVC. Can be templated. + # existingClaim: + + ## If persistence is not enabled, this allows to mount the + ## local storage in-memory to improve performance + ## + inMemory: + enabled: false + ## The maximum usage on memory medium EmptyDir would be + ## the minimum value between the SizeLimit specified + ## here and the sum of memory limits of all containers in a pod + ## + # sizeLimit: 300Mi + +initChownData: + ## If false, data ownership will not be reset at startup + ## This allows the prometheus-server to be run with an arbitrary user + ## + enabled: true + + ## initChownData container image + ## + image: + registry: docker.io + repository: busybox + tag: "1.31.1" + sha: "" + pullPolicy: IfNotPresent + + ## initChownData resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 200m + memory: 2568Mi + requests: + cpu: 100m + memory: 128Mi + + +# Administrator credentials when not using an existing secret (see below) +adminUser: admin +# adminPassword + +# Use an existing secret for the admin user. +admin: + ## Name of the secret. Can be templated. + existingSecret: "" + userKey: admin-user + passwordKey: admin-password + +## Define command to be executed at startup by grafana container +## Needed if using `vault-env` to manage secrets (ref: https://banzaicloud.com/blog/inject-secrets-into-pods-vault/) +## Default is "run.sh" as defined in grafana's Dockerfile +# command: +# - "sh" +# - "/run.sh" + +## Use an alternate scheduler, e.g. "stork". +## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ +## +# schedulerName: + +## Extra environment variables that will be pass onto deployment pods +## +## to provide grafana with access to CloudWatch on AWS EKS: +## 1. create an iam role of type "Web identity" with provider oidc.eks.* (note the provider for later) +## 2. edit the "Trust relationships" of the role, add a line inside the StringEquals clause using the +## same oidc eks provider as noted before (same as the existing line) +## also, replace NAMESPACE and prometheus-operator-grafana with the service account namespace and name +## +## "oidc.eks.us-east-1.amazonaws.com/id/XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX:sub": "system:serviceaccount:NAMESPACE:prometheus-operator-grafana", +## +## 3. attach a policy to the role, you can use a built in policy called CloudWatchReadOnlyAccess +## 4. use the following env: (replace 123456789000 and iam-role-name-here with your aws account number and role name) +## +## env: +## AWS_ROLE_ARN: arn:aws:iam::123456789000:role/iam-role-name-here +## AWS_WEB_IDENTITY_TOKEN_FILE: /var/run/secrets/eks.amazonaws.com/serviceaccount/token +## AWS_REGION: us-east-1 +## +## 5. uncomment the EKS section in extraSecretMounts: below +## 6. uncomment the annotation section in the serviceAccount: above +## make sure to replace arn:aws:iam::123456789000:role/iam-role-name-here with your role arn + +env: {} + +## "valueFrom" environment variable references that will be added to deployment pods. Name is templated. +## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#envvarsource-v1-core +## Renders in container spec as: +## env: +## ... +## - name: +## valueFrom: +## +envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + +## The name of a secret in the same kubernetes namespace which contain values to be added to the environment +## This can be useful for auth tokens, etc. Value is templated. +envFromSecret: "" + +## Sensible environment variables that will be rendered as new secret object +## This can be useful for auth tokens, etc +envRenderSecret: {} + +## The names of secrets in the same kubernetes namespace which contain values to be added to the environment +## Each entry should contain a name key, and can optionally specify whether the secret must be defined with an optional key. +## Name is templated. +envFromSecrets: [] +## - name: secret-name +## optional: true + +## The names of conifgmaps in the same kubernetes namespace which contain values to be added to the environment +## Each entry should contain a name key, and can optionally specify whether the configmap must be defined with an optional key. +## Name is templated. +## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.23/#configmapenvsource-v1-core +envFromConfigMaps: [] +## - name: configmap-name +## optional: true + +# Inject Kubernetes services as environment variables. +# See https://kubernetes.io/docs/concepts/services-networking/connect-applications-service/#environment-variables +enableServiceLinks: true + +## Additional grafana server secret mounts +# Defines additional mounts with secrets. Secrets must be manually created in the namespace. +extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # secretName: grafana-secret-files + # readOnly: true + # subPath: "" + # + # for AWS EKS (cloudwatch) use the following (see also instruction in env: above) + # - name: aws-iam-token + # mountPath: /var/run/secrets/eks.amazonaws.com/serviceaccount + # readOnly: true + # projected: + # defaultMode: 420 + # sources: + # - serviceAccountToken: + # audience: sts.amazonaws.com + # expirationSeconds: 86400 + # path: token + # + # for CSI e.g. Azure Key Vault use the following + # - name: secrets-store-inline + # mountPath: /run/secrets + # readOnly: true + # csi: + # driver: secrets-store.csi.k8s.io + # readOnly: true + # volumeAttributes: + # secretProviderClass: "akv-grafana-spc" + # nodePublishSecretRef: # Only required when using service principal mode + # name: grafana-akv-creds # Only required when using service principal mode + +## Additional grafana server volume mounts +# Defines additional volume mounts. +extraVolumeMounts: [] + # - name: extra-volume-0 + # mountPath: /mnt/volume0 + # readOnly: true + # existingClaim: volume-claim + # - name: extra-volume-1 + # mountPath: /mnt/volume1 + # readOnly: true + # hostPath: /usr/shared/ + +## Container Lifecycle Hooks. Execute a specific bash command or make an HTTP request +lifecycleHooks: {} + # postStart: + # exec: + # command: [] + +## Pass the plugins you want installed as a list. +## +plugins: + - grafana-piechart-panel + # - digrich-bubblechart-panel + # - grafana-clock-panel + +## Configure grafana datasources +## ref: http://docs.grafana.org/administration/provisioning/#datasources +## +datasources: {} +# datasources.yaml: +# apiVersion: 1 +# datasources: +# - name: Prometheus +# type: prometheus +# url: http://prometheus-prometheus-server +# access: proxy +# isDefault: true +# - name: CloudWatch +# type: cloudwatch +# access: proxy +# uid: cloudwatch +# editable: false +# jsonData: +# authType: default +# defaultRegion: us-east-1 + +## Configure notifiers +## ref: http://docs.grafana.org/administration/provisioning/#alert-notification-channels +## +notifiers: {} +# notifiers.yaml: +# notifiers: +# - name: email-notifier +# type: email +# uid: email1 +# # either: +# org_id: 1 +# # or +# org_name: Main Org. +# is_default: true +# settings: +# addresses: an_email_address@example.com +# delete_notifiers: + +## Configure grafana dashboard providers +## ref: http://docs.grafana.org/administration/provisioning/#dashboards +## +## `path` must be /var/lib/grafana/dashboards/ +## +dashboardProviders: {} +# dashboardproviders.yaml: +# apiVersion: 1 +# providers: +# - name: 'default' +# orgId: 1 +# folder: '' +# type: file +# disableDeletion: false +# editable: true +# options: +# path: /var/lib/grafana/dashboards/default + +## Configure grafana dashboard to import +## NOTE: To use dashboards you must also enable/configure dashboardProviders +## ref: https://grafana.com/dashboards +## +## dashboards per provider, use provider name as key. +## +dashboards: + default: + logging-dashboard: + file: dashboards/logging-dashboard.json + backup-detail: + file: dashboards/backup-detail.json + clusterbackup-detail: + file: dashboards/clusterbackup-detail.json + backup-overview: + file: dashboards/backup-overview.json + backupplan-detail: + file: dashboards/backupplan-detail.json + clusterbackupplan-detail: + file: dashboards/clusterbackupplan-detail.json + backupplan-overview: + file: dashboards/backupplan-overview.json + metadata-detail: + file: dashboards/metadata-detail.json + overview: + file: dashboards/overview.json + restore-detail: + file: dashboards/restore-detail.json + clusterrestore-detail: + file: dashboards/clusterrestore-detail.json + restore-overview: + file: dashboards/restore-overview.json + target-detail: + file: dashboards/target-detail.json + continuousrestoreplan-detail: + file: dashboards/continuousrestoreplan-detail.json + consistentset-detail: + file: dashboards/consistentset-detail.json + # default: + # some-dashboard: + # json: | + # $RAW_JSON + # custom-dashboard: + # file: dashboards/custom-dashboard.json + # prometheus-stats: + # gnetId: 2 + # revision: 2 + # datasource: Prometheus + # local-dashboard: + # url: https://example.com/repository/test.json + # token: '' + # local-dashboard-base64: + # url: https://example.com/repository/test-b64.json + # token: '' + # b64content: true + +## Reference to external ConfigMap per provider. Use provider name as key and ConfigMap name as value. +## A provider dashboards must be defined either by external ConfigMaps or in values.yaml, not in both. +## ConfigMap data example: +## +## data: +## example-dashboard.json: | +## RAW_JSON +## +dashboardsConfigMaps: {} +# default: "" + +## Grafana's primary configuration +## NOTE: values in map will be converted to ini format +## ref: http://docs.grafana.org/installation/configuration/ +## +grafana.ini: + dashboards: + default_home_dashboard_path: /var/lib/grafana/dashboards/default/overview.json + paths: + data: /var/lib/grafana/ + logs: /var/log/grafana + plugins: /var/lib/grafana/plugins + provisioning: /etc/grafana/provisioning + analytics: + check_for_updates: false + log: + mode: console + server: + root_url: "%(protocol)s://%(domain)s:%(http_port)s/grafana/" + serve_from_sub_path: true +## grafana Authentication can be enabled with the following values on grafana.ini +# server: + # The full public facing url you use in browser, used for redirects and emails + # root_url: + # https://grafana.com/docs/grafana/latest/auth/github/#enable-github-in-grafana + # auth.github: + # enabled: false + # allow_sign_up: false + # scopes: user:email,read:org + # auth_url: https://github.com/login/oauth/authorize + # token_url: https://github.com/login/oauth/access_token + # api_url: https://api.github.com/user + # team_ids: + # allowed_organizations: + # client_id: + # client_secret: +## LDAP Authentication can be enabled with the following values on grafana.ini +## NOTE: Grafana will fail to start if the value for ldap.toml is invalid + # auth.ldap: + # enabled: true + # allow_sign_up: true + # config_file: /etc/grafana/ldap.toml + +## Grafana's LDAP configuration +## Templated by the template in _helpers.tpl +## NOTE: To enable the grafana.ini must be configured with auth.ldap.enabled +## ref: http://docs.grafana.org/installation/configuration/#auth-ldap +## ref: http://docs.grafana.org/installation/ldap/#configuration +ldap: + enabled: false + # `existingSecret` is a reference to an existing secret containing the ldap configuration + # for Grafana in a key `ldap-toml`. + existingSecret: "" + # `config` is the content of `ldap.toml` that will be stored in the created secret + config: "" + # config: |- + # verbose_logging = true + + # [[servers]] + # host = "my-ldap-server" + # port = 636 + # use_ssl = true + # start_tls = false + # ssl_skip_verify = false + # bind_dn = "uid=%s,ou=users,dc=myorg,dc=com" + +## Grafana's SMTP configuration +## NOTE: To enable, grafana.ini must be configured with smtp.enabled +## ref: http://docs.grafana.org/installation/configuration/#smtp +smtp: + # `existingSecret` is a reference to an existing secret containing the smtp configuration + # for Grafana. + existingSecret: "" + userKey: "user" + passwordKey: "password" + +## Sidecars that collect the configmaps with specified label and stores the included files them into the respective folders +## Requires at least Grafana 5 to work and can't be used together with parameters dashboardProviders, datasources and dashboards +sidecar: + image: + registry: quay.io + repository: kiwigrid/k8s-sidecar + tag: 1.15.6 + sha: "" + imagePullPolicy: IfNotPresent + resources: + limits: + cpu: 200m + memory: 200Mi + requests: + cpu: 100m + memory: 100Mi + securityContext: {} + # skipTlsVerify Set to true to skip tls verification for kube api calls + # skipTlsVerify: true + enableUniqueFilenames: false + readinessProbe: {} + livenessProbe: {} + dashboards: + enabled: true + SCProvider: true + # label that the configmaps with dashboards are marked with + label: grafana_dashboard + # value of label that the configmaps with dashboards are set to + labelValue: null + # folder in the pod that should hold the collected dashboards (unless `defaultFolderName` is set) + folder: /tmp/dashboards + # The default folder name, it will create a subfolder under the `folder` and put dashboards in there instead + defaultFolderName: null + # Namespaces list. If specified, the sidecar will search for config-maps/secrets inside these namespaces. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces. + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # If specified, the sidecar will look for annotation with this name to create folder and put graph here. + # You can use this parameter together with `provider.foldersFromFilesStructure`to annotate configmaps and create folder structure. + folderAnnotation: null + # Absolute path to shell script to execute after a configmap got reloaded + script: null + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # provider configuration that lets grafana manage the dashboards + provider: + # name of the provider, should be unique + name: sidecarProvider + # orgid as configured in grafana + orgid: 1 + # folder in which the dashboards should be imported in grafana + folder: '' + # type of the provider + type: file + # disableDelete to activate a import-only behaviour + disableDelete: false + # allow updating provisioned dashboards from the UI + allowUiUpdates: false + # allow Grafana to replicate dashboard structure from filesystem + foldersFromFilesStructure: false + # Additional dashboard sidecar volume mounts + extraMounts: [] + # Sets the size limit of the dashboard sidecar emptyDir volume + sizeLimit: {} + datasources: + enabled: true + # label that the configmaps with datasources are marked with + label: grafana_datasource + # value of label that the configmaps with datasources are set to + labelValue: null + # If specified, the sidecar will search for datasource config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # Endpoint to send request to reload datasources + reloadURL: "http://localhost:3000/api/admin/provisioning/datasources/reload" + skipReload: false + # Deploy the datasource sidecar as an initContainer in addition to a container. + # This is needed if skipReload is true, to load any datasources defined at startup time. + initDatasources: false + # Sets the size limit of the datasource sidecar emptyDir volume + sizeLimit: {} + plugins: + enabled: false + # label that the configmaps with plugins are marked with + label: grafana_plugin + # value of label that the configmaps with plugins are set to + labelValue: null + # If specified, the sidecar will search for plugin config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # Endpoint to send request to reload plugins + reloadURL: "http://localhost:3000/api/admin/provisioning/plugins/reload" + skipReload: false + # Deploy the datasource sidecar as an initContainer in addition to a container. + # This is needed if skipReload is true, to load any plugins defined at startup time. + initPlugins: false + # Sets the size limit of the plugin sidecar emptyDir volume + sizeLimit: {} + notifiers: + enabled: false + # label that the configmaps with notifiers are marked with + label: grafana_notifier + # If specified, the sidecar will search for notifier config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # search in configmap, secret or both + resource: both + # Sets the size limit of the notifier sidecar emptyDir volume + sizeLimit: {} + +## Override the deployment namespace +## +namespaceOverride: "" + +## Number of old ReplicaSets to retain +## +revisionHistoryLimit: 10 + +## Add a seperate remote image renderer deployment/service +imageRenderer: + # Enable the image-renderer deployment & service + enabled: false + replicas: 1 + image: + registry: docker.io + # image-renderer Image repository + repository: grafana/grafana-image-renderer + # image-renderer Image tag + tag: latest + # image-renderer Image sha (optional) + sha: "" + # image-renderer ImagePullPolicy + pullPolicy: Always + # extra environment variables + env: + HTTP_HOST: "0.0.0.0" + # RENDERING_ARGS: --no-sandbox,--disable-gpu,--window-size=1280x758 + # RENDERING_MODE: clustered + # IGNORE_HTTPS_ERRORS: true + # image-renderer deployment serviceAccount + serviceAccountName: "" + # image-renderer deployment securityContext + securityContext: {} + # image-renderer deployment Host Aliases + hostAliases: [] + # image-renderer deployment priority class + priorityClassName: '' + service: + # Enable the image-renderer service + enabled: true + # image-renderer service port name + portName: 'http' + # image-renderer service port used by both service and deployment + port: 8081 + targetPort: 8081 + # If https is enabled in Grafana, this needs to be set as 'https' to correctly configure the callback used in Grafana + grafanaProtocol: http + # In case a sub_path is used this needs to be added to the image renderer callback + grafanaSubPath: "" + # name of the image-renderer port on the pod + podPortName: http + # number of image-renderer replica sets to keep + revisionHistoryLimit: 10 + networkPolicy: + # Enable a NetworkPolicy to limit inbound traffic to only the created grafana pods + limitIngress: true + # Enable a NetworkPolicy to limit outbound traffic to only the created grafana pods + limitEgress: false + resources: + limits: + cpu: 200m + memory: 200Mi + requests: + cpu: 100m + memory: 100Mi + ## Node labels for pod assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + # + nodeSelector: {} + + ## Tolerations for pod assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + + ## Affinity for pod assignment (evaluated as template) + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## + affinity: {} + +networkPolicy: + ## @param networkPolicy.enabled Enable creation of NetworkPolicy resources. Only Ingress traffic is filtered for now. + ## + enabled: false + ## @param networkPolicy.allowExternal Don't require client label for connections + ## The Policy model to apply. When set to false, only pods with the correct + ## client label will have network access to grafana port defined. + ## When true, grafana will accept connections from any source + ## (with the correct destination port). + ## + allowExternal: true + ## @param networkPolicy.explicitNamespacesSelector A Kubernetes LabelSelector to explicitly select namespaces from which traffic could be allowed + ## If explicitNamespacesSelector is missing or set to {}, only client Pods that are in the networkPolicy's namespace + ## and that match other criteria, the ones that have the good label, can reach the grafana. + ## But sometimes, we want the grafana to be accessible to clients from other namespaces, in this case, we can use this + ## LabelSelector to select these namespaces, note that the networkPolicy's namespace should also be explicitly added. + ## + ## Example: + ## explicitNamespacesSelector: + ## matchLabels: + ## role: frontend + ## matchExpressions: + ## - {key: role, operator: In, values: [frontend]} + ## + explicitNamespacesSelector: {} + +# Enable backward compatibility of kubernetes where version below 1.13 doesn't have the enableServiceLinks option +enableKubeBackwardCompatibility: false + +# Create a dynamic manifests via values: +extraObjects: [] + # - apiVersion: "kubernetes-client.io/v1" + # kind: ExternalSecret + # metadata: + # name: grafana-secrets + # spec: + # backendType: gcpSecretsManager + # data: + # - key: grafana-admin-password + # name: adminPassword diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/crds/triliovault.trilio.io_triliovaultmanagers.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/crds/triliovault.trilio.io_triliovaultmanagers.yaml new file mode 100644 index 000000000..26ebc28eb --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/crds/triliovault.trilio.io_triliovaultmanagers.yaml @@ -0,0 +1,1231 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.10.0 + creationTimestamp: null + name: triliovaultmanagers.triliovault.trilio.io +spec: + group: triliovault.trilio.io + names: + kind: TrilioVaultManager + listKind: TrilioVaultManagerList + plural: triliovaultmanagers + shortNames: + - tvm + singular: triliovaultmanager + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.releaseVersion + name: TrilioVault-Version + type: string + - jsonPath: .spec.applicationScope + name: Scope + type: string + - jsonPath: .status.status + name: Status + type: string + name: v1 + schema: + openAPIV3Schema: + description: TrilioVaultManager is the Schema for the triliovaultmanagers + 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: TrilioVaultManagerSpec defines the desired state of TrilioVaultManager + properties: + affinity: + description: The scheduling constraints on application pods. + 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 + required: + - key + - operator + type: object + type: array + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + required: + - key + - operator + type: object + type: array + 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 + required: + - key + - operator + type: object + type: array + type: object + x-kubernetes-map-type: atomic + type: array + 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. + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + 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. + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + 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. + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + 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. + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + required: + - key + - operator + type: object + type: array + 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 + 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 + type: object + type: object + applicationScope: + description: Scope for the application which will be installed in + the cluster NamespaceScope or ClusterScope + enum: + - Cluster + - Namespaced + type: string + componentConfiguration: + description: ComponentConfiguration holds all the field related to + the TVK deployments. + properties: + admission-webhook: + description: AdmissionWebhook holds all configuration for the + admission-webhook deployment + type: object + x-kubernetes-preserve-unknown-fields: true + control-plane: + description: ControlPlane holds all configuration for the control-plane + deployment + type: object + x-kubernetes-preserve-unknown-fields: true + exporter: + description: Exporter holds all configuration for the exporter + deployment. + type: object + x-kubernetes-preserve-unknown-fields: true + ingress-controller: + description: IngressController holds all configuration for the + ingress-controller deployment + type: object + x-kubernetes-preserve-unknown-fields: true + web: + description: Web holds all configuration for the web deployment + type: object + x-kubernetes-preserve-unknown-fields: true + web-backend: + description: WebBackend holds all configuration for the web-backend + deployment + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + csiConfig: + description: CSIConfig is the configuration for the CSI drivers which + doesn't support snapshot functionality + properties: + exclude: + description: Exclude denotes the list of CSI drivers to be excluded + from the non-snapshot functionality category + items: + type: string + type: array + include: + description: Include denotes the list of CSI drivers to be included + in the non-snapshot functionality category + items: + type: string + type: array + type: object + dataJobLimits: + 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: 'Deprecated: DataJobLimits are the resource limits for + all the data processing jobs.' + type: object + dataJobResources: + description: DataJobResources is the resource limits & requests for + all the data processing jobs. + 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 + description: 'Limits describes the maximum amount of compute resources + allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + 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 + 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. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + type: object + datamoverLogLevel: + description: DatamoverLogLevel is a log level used in datamover i.e. + data upload/restore part of the TVK. + enum: + - Panic + - Fatal + - Error + - Warn + - Info + - Debug + - Trace + type: string + deploymentLimits: + 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: 'Deprecated: DeploymentLimits are the resource limits + for all the deployments.' + type: object + helmValues: + description: HelmValues holds all the additional fields in the values.yaml + of TVK helm chart. + type: object + x-kubernetes-preserve-unknown-fields: true + helmVersion: + description: 'Deprecated: Helm Version' + properties: + tillerNamespace: + type: string + version: + enum: + - v3 + type: string + required: + - version + type: object + ingressConfig: + description: IngressConfig holds field related to ingress resource + to access the TVK UI. + properties: + annotations: + additionalProperties: + type: string + type: object + host: + type: string + ingressClass: + type: string + tlsSecretName: + type: string + type: object + logLevel: + description: LogLevel is a level used in TVK logging. + enum: + - Panic + - Fatal + - Error + - Warn + - Info + - Debug + - Trace + type: string + metadataJobLimits: + 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: 'Deprecated: MetadataJobLimits are the resource limits + for all the meta processing jobs.' + type: object + metadataJobResources: + description: MetadataJobResources is the resource limits & requests + for all the meta processing jobs. + 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 + description: 'Limits describes the maximum amount of compute resources + allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + 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 + 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. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + type: object + nodeSelector: + additionalProperties: + type: string + description: NodeSelector specifies a map of key-value pairs. For + the pod to be eligible to run on a node, the node must have each + of the indicated key-value pairs as labels. + type: object + pauseSchedule: + description: PauseSchedule is flag to pause schedule backups or snapshot + for all the backupplan/clusterbackupplan. + type: boolean + resources: + description: 'Deprecated: Resources are the resource requirements + for the containers.' + 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 + description: 'Limits describes the maximum amount of compute resources + allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + 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 + 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. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + type: object + restoreNamespaces: + description: 'Deprecated: RestoreNamespaces are the namespace where + you want to restore your applications. Restore Namespaces depends + on your k8s RBAC' + items: + type: string + type: array + tolerations: + description: The toleration of application against the specific taints + on the nodes + 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 + trilioVaultAppVersion: + description: 'Deprecated: TrilioVaultAppVersion Helm Chart version' + type: string + tvkInstanceName: + description: TVKInstanceName is a TVK installation name to be displayed + on UI. + type: string + required: + - applicationScope + type: object + status: + description: TrilioVaultManagerStatus defines the observed state of TrilioVaultManager + properties: + conditions: + items: + properties: + lastTransitionTime: + format: date-time + nullable: true + type: string + message: + minLength: 0 + type: string + reason: + enum: + - InstallSuccessful + - UpdateSuccessful + - UninstallSuccessful + - InstallError + - UpdateError + - ReconcileError + - UninstallError + type: string + status: + enum: + - "True" + - "False" + - Unknown + type: string + type: + enum: + - Initialized + - Deployed + - Updated + - ReleaseFailed + - Irreconcilable + type: string + type: object + type: array + dashboard: + type: string + deployedRelease: + properties: + manifest: + type: string + name: + type: string + type: object + helmRevision: + type: integer + releaseVersion: + type: string + status: + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/questions.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/questions.yaml new file mode 100644 index 000000000..c5064a13f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/questions.yaml @@ -0,0 +1,158 @@ +questions: +- variable: installTVK.enabled + default: true + description: "TriloVault Manager is an instance of TrilioVault for Kubernetes. Selecting this checkbox automatically creates a TrilioVault Manager instance" + required: true + type: boolean + label: "Install TrilioVault Manager Automatically" + group: "TrilioVault Manager Install Configuration" + +- variable: installTVK.tvkInstanceName + show_if: "installTVK.enabled=true" + default: "triliovault-manager" + description: "TrilioVault Manager Instance Name. This will be used to manage the Kubernetes cluster in TVK Management Console and backups performed by the TrilioVault for Kubernetes" + required: true + type: string + label: "TrilioVault Manager Instance Name" + group: "TrilioVault Manager Install Configuration" + +- variable: installTVK.applicationScope + default: Cluster + description: "TrilioVault Manager installation scope: Cluster or Namespaced" + required: true + type: enum + label: "TrilioVault Manager Installation Scope" + group: "TrilioVault Manager Install Configuration" + options: + - "Cluster" + - "Namespaced" + +- variable: installTVK.ingressConfig.host + default: "rancher.k8s-tvk.com" + description: "Hostname URL to access the TVK Management Console - For example: rancher.k8s-tvk.com" + required: true + type: hostname + label: "TVK Management Console Hostname URL" + group: "Ingress Configuration" + +- variable: installTVK.ingressConfig.tlsSecretName + default: "" + description: "TLS Secret containing an appropriate certificate to access the TVK Management Console over HTTPS protocol. Secret should of type kubernetes.io/tls" + required: false + type: secret + label: "TLS Secret of type kubernetes.io/tls (Optional)" + group: "Ingress Configuration" + + +- variable: installTVK.ComponentConfiguration.ingressController.service.type + default: "NodePort" + description: "Ingress Controller Service Type to access the TVK Management Console" + required: true + type: enum + label: "Ingress Controller Service Type" + group: "Ingress Configuration" + options: + - "NodePort" + - "LoadBalancer" + +- variable: installTVK.ingressConfig.annotations + default: "" + description: "Annotations to add for the TrilioVault Manager ingress resource - For example: {'foo':'bar'}" + required: false + type: string + label: "Annotations for Ingress Resource (Optional)" + group: "Ingress Configuration" + +- variable: proxySettings.PROXY_ENABLED + default: false + description: "Select this checkbox to deploy the TrilioVault Manager via a proxy server" + required: false + type: boolean + label: "Proxy Settings (Optional)" + group: "Proxy Settings" + show_subquestion_if: true + subquestions: + - variable: proxySettings.NO_PROXY + default: "" + description: "Provide the user defined IPs/hosts and subnets to exempt from proxy. User can provide comma separated values. For example: 'localhost,127.0.0.1,10.239.112.0/20,10.240.0.0/14'" + required: false + type: string + label: "No Proxy (Optional)" + group: "Proxy Settings" + - variable: proxySettings.HTTP_PROXY + default: "" + description: "Provide HTTP proxy information. For example: http://:@:" + required: true + type: string + label: "HTTP Proxy" + group: "Proxy Settings" + - variable: proxySettings.HTTPS_PROXY + default: "" + description: "Provide HTTPS proxy information. For example: https://:@:" + required: true + type: string + label: "HTTPS Proxy" + group: "Proxy Settings" + - variable: proxySettings.CA_BUNDLE_CONFIGMAP + default: "" + description: "Provide a CA Certificate bundle configmap present on the Kubernetes cluster to communicate with the proxy server" + required: false + type: string + label: "CA Certificate Bundle Configmap Name (Optional)" + group: "Proxy Settings" + +- variable: observability.enabled + default: false + description: "Select this checkbox to deploy the Observability Stack with Triliovault operator" + required: false + type: boolean + label: "Observability Stack (Optional)" + group: "Observability" + show_subquestion_if: true + subquestions: + - variable: observability.logging.loki.enabled + default: true + description: "Select this checkbox to deploy the Logging Stack with Loki" + required: true + type: boolean + label: "Logging with Loki" + group: "Logging" + - variable: observability.logging.promtail.enabled + default: true + description: "Select this checkbox to deploy the Logging Stack with Promtail" + required: true + type: boolean + label: "Logging with Promtail" + group: "Logging" + - variable: observability.monitoring.prometheus.enabled + default: true + description: "Select this checkbox to deploy the Monitoring Stack with Prometheus" + required: true + type: boolean + label: "Monitoring with Prometheus" + group: "Monitoring" + - variable: observability.monitoring.prometheus.server.enabled + default: true + description: "Select this checkbox to deploy the Monitoring Stack with Prometheus Server" + required: true + type: boolean + label: "Monitoring with Prometheus Server" + group: "Monitoring" + - variable: observability.visualization.grafana.enabled + default: true + description: "Select this checkbox to deploy the Visualization Stack with Grafana" + required: true + type: boolean + label: "Visualization with Grafana" + group: "Visualization" + - variable: observability.visualization.grafana.service.type + show_if: "observability.visualization.grafana.enabled=true" + default: "ClusterIP" + description: "Grafana Service Type to access the Grafana Dashboards" + required: true + type: enum + label: "Grafana Service Type" + group: "Visualization" + options: + - "NodePort" + - "LoadBalancer" diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/NOTES.txt new file mode 100644 index 000000000..587919034 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/NOTES.txt @@ -0,0 +1,59 @@ +TrilioVault Operator is a helm based operator which install/upgrade/delete the helm Chart of the TrilioVault For Kubernetes. +This operator watches over the entire helm application of TrilioVault for Kubernetes and has self-healing capabilities. + +To verify that TrilioVault Operator has started, run: + + kubectl --namespace={{ .Release.Namespace }} wait --for=condition=ready pod -l "release={{ .Release.Name }}" + +{{ if .Values.installTVK.enabled }} +In one click install, a cluster scope TVM custom resource triliovault-manager is created, you can check its +configuration by running following command: + + kubectl --namespace {{ .Release.Namespace }} get triliovaultmanagers.triliovault.trilio.io triliovault-manager -o yaml + +{{- else }} + +Once the Triliovault operator is in running state, you can create the TrilioVault for Kubernetes(TVK) with the +following custom resource: + + apiVersion: triliovault.trilio.io/v1 + kind: TrilioVaultManager + metadata: + labels: + app: triliovault + name: triliovault-manager + namespace: {{ .Release.Namespace }} + spec: + trilioVaultAppVersion: latest + applicationScope: Cluster + ingressConfig: + host: "" + componentConfiguration: + ingress-controller: + enabled: true + service: + type: LoadBalancer + +Once the above CR has been created, you have to wait for the TVK pods to come up. +{{- end }} + +To check all the TVK pods come into running state, run: + + kubectl --namespace {{ .Release.Namespace }} wait --for=condition=ready pod -l "release=triliovault-manager-{{ .Release.Namespace }}" + +Once all the pods are in running state, you can access the TVK UI from your browser using following steps: + +{{- if .Values.installTVK.enabled }} +{{- if eq .Values.installTVK.ComponentConfiguration.ingressController.service.type "LoadBalancer" }} + 1. Find the external IP of the service `k8s-triliovault-ingress-nginx-controller` + 2. Hit the URL in browser: https:///t4k/ +{{- else }} + 1. Find the NodePort from the service `k8s-triliovault-ingress-nginx-controller` + 2. Hit the URL in browser with NodePort: https://:/t4k/ +{{- end }} +{{- end }} + +For more details on how to access the TVK UI, follow this guide: https://docs.trilio.io/kubernetes/management-console-ui/accessing-the-ui + +You can start backup and restore of your application using TVK. For more details on how to do that, please follow our +getting started guide: https://docs.trilio.io/kubernetes/advanced-configuration/management-console diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMCustomResource.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMCustomResource.yaml new file mode 100644 index 000000000..e3a188388 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMCustomResource.yaml @@ -0,0 +1,65 @@ +{{- if .Values.installTVK.enabled }} +{{- if not (lookup "triliovault.trilio.io/v1" "TrilioVaultManager" "" "").items }} + {{template "k8s-triliovault-operator.tlsSecretValidation" .}} +apiVersion: triliovault.trilio.io/v1 +kind: TrilioVaultManager +metadata: + name: "triliovault-manager" + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install +spec: + applicationScope: {{ .Values.installTVK.applicationScope }} + {{- if .Values.installTVK.tvkInstanceName }} + tvkInstanceName: {{ .Values.installTVK.tvkInstanceName }} + {{- end }} + {{- if or .Values.imagePullSecret .Values.svcAccountName .Values.observability.enabled .Values.global.urlPath }} + helmValues: + urlPath: {{ .Values.global.urlPath | quote }} + {{- if .Values.observability.enabled }} + observability: + name: {{ .Values.observability.name }} + namespace: {{ default .Release.Namespace }} + {{- end }} + {{- if include "k8s-triliovault-operator.imagePullSecret" . }} + imagePullSecret: {{ template "k8s-triliovault-operator.imagePullSecret" . }} + {{- end }} + {{- if .Values.svcAccountName }} + svcAccountName: {{ .Values.svcAccountName }} + {{- end }} + {{- end }} + {{- if .Values.nodeSelector }} + nodeSelector: {{- .Values.nodeSelector | toYaml | nindent 4 }} + {{- end }} + {{- if .Values.affinity }} + affinity: + {{- toYaml .Values.affinity | nindent 4 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 4 }} + {{- end }} + # User can configure the ingress hosts, annotations and TLS secret through the ingressConfig section + ingressConfig: + {{- if and (gt (len .Values.installTVK.ingressConfig.annotations) 0) (not .Values.installTVK.ComponentConfiguration.ingressController.enabled) }} + annotations: + {{- range $key, $value := .Values.installTVK.ingressConfig.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end -}} + {{- end }} + host: {{ .Values.installTVK.ingressConfig.host | quote }} + {{- if not .Values.installTVK.ComponentConfiguration.ingressController.enabled }} + ingressClass: {{ .Values.installTVK.ingressConfig.ingressClass | quote }} + {{- end }} + {{- if .Values.installTVK.ingressConfig.tlsSecretName }} + tlsSecretName: {{ .Values.installTVK.ingressConfig.tlsSecretName | quote }} + {{- end }} + # TVK components configuration, currently supports control-plane, web, exporter, web-backend, ingress-controller, admission-webhook. + # User can configure resources for all componentes and can configure service type and host for the ingress-controller + componentConfiguration: + ingress-controller: + enabled: {{ .Values.installTVK.ComponentConfiguration.ingressController.enabled }} + service: + type: {{ .Values.installTVK.ComponentConfiguration.ingressController.service.type }} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMSecret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMSecret.yaml new file mode 100644 index 000000000..0d9d8a9df --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/TVMSecret.yaml @@ -0,0 +1,25 @@ +{{- if .Values.observability.enabled }} +apiVersion: v1 +kind: Secret +metadata: + name: tvk-integration + namespace: {{ .Release.Namespace }} + annotations: + meta.helm.sh/release-namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + triliovault.trilio.io/owner: {{ template "k8s-triliovault-operator.appName" . }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-validation-webhook-configuration + triliovault.trilio.io/observability: "true" +type: Opaque +stringData: + integration: |- + type: Loki + protocol: "" + host: "" + port: "" + path: "/api/v1/datasource" + username: "admin" + password: {{ .Values.observability.visualization.grafana.adminPassword | quote }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/_helpers.tpl new file mode 100644 index 000000000..120ef9c7d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/_helpers.tpl @@ -0,0 +1,134 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "k8s-triliovault-operator.name" -}} +{{- default .Release.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{- define "k8s-triliovault-operator.appName" -}} +{{- printf "%s" .Chart.Name -}} +{{- 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 "k8s-triliovault-operator.fullname" -}} +{{- printf "%s" .Chart.Name -}} +{{- end -}} + +{{/* +Return the proper TrilioVault Operator image name +*/}} +{{- define "k8s-triliovault-operator.image" -}} +{{- $registryName := .Values.image.registry -}} +{{- $repositoryName := .Values.image.repository -}} +{{- $tag := .Values.image.tag | toString -}} +{{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} +{{- end -}} + +{{/* +Validation of the secret of CA bundle if provided +*/}} +{{- define "k8s-triliovault-operator.caBundleValidation" -}} +{{- if .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} +{{- if not (lookup "v1" "ConfigMap" .Release.Namespace .Values.proxySettings.CA_BUNDLE_CONFIGMAP) }} + {{ fail "Proxy CA bundle proxy is not present in the release namespace" }} +{{- else }} + {{- $caMap := (lookup "v1" "ConfigMap" .Release.Namespace .Values.proxySettings.CA_BUNDLE_CONFIGMAP).data }} + {{- if not (get $caMap "ca-bundle.crt") }} + {{ fail "Proxy CA certificate file key should be ca-bundle.crt" }} + {{- end }} +{{- end }} +{{- end }} +{{- end -}} + +{{/* +Validation for the ingress tlsSecret, should exists if provided +*/}} + +{{- define "k8s-triliovault-operator.tlsSecretValidation" }} +{{- if .Values.installTVK.ingressConfig.tlsSecretName -}} +{{- if not (lookup "v1" "Secret" .Release.Namespace .Values.installTVK.ingressConfig.tlsSecretName ) -}} + {{ fail "Ingress tls secret is not present in the release namespace" }} +{{- end -}} +{{- end -}} +{{- end -}} + + +{{- define "k8s-triliovault-operator.preFlightValidation" }} +{{- if not .Values.preflight.storageClass }} + {{ fail "Provide the name of storage class as you have enabled the preflight" }} +{{- else }} + {{- if not (lookup "storage.k8s.io/v1" "StorageClass" "" .Values.preflight.storageClass) }} + {{ fail "Storage class provided is not present in the cluster" }} + {{- end }} +{{- end }} +{{- end }} + +{{- define "k8s-triliovault-operator.priorityClassValidator" }} +{{- if .Values.priorityClassName -}} +{{- if not (lookup "scheduling.k8s.io/v1" "PriorityClass" "" .Values.priorityClassName) }} + {{ fail "Priority class provided is not present in the cluster" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create unified labels for k8s-triliovault-operator components +*/}} +{{- define "k8s-triliovault-operator.labels" -}} +app.kubernetes.io/part-of: {{ template "k8s-triliovault-operator.appName" . }} +app.kubernetes.io/name: {{ template "k8s-triliovault-operator.appName" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} + +{{- define "k8s-triliovault-operator.serviceAccountName" -}} + {{- if eq .Values.svcAccountName "" -}} + {{- printf "%s" "k8s-triliovault-operator-service-account" -}} + {{- else -}} + {{- printf "%s" .Values.svcAccountName -}} + {{- end -}} +{{- end -}} + +{{- define "k8s-triliovault-operator.preflightServiceAccountName" -}} + {{- if eq .Values.svcAccountName "" -}} + {{- printf "%s" "k8s-triliovault-operator-preflight-service-account" -}} + {{- else -}} + {{- printf "%s" .Values.svcAccountName -}} + {{- end -}} +{{- end -}} + +{{/* +Return the imagePullSecret name in below priority order +1. Returns imagePullSecret name if imagePullSecret is supplied via helm value during installation time +2. If the helm value is not provided and a service account name is provided via svcAccountName parameter, this extracts and returns imagePullSecret from service account if available. + (In case of multiple imagePullSecrets are attached to a service account, only the first one is taken into the consideration) +3. Returns empty string not imagePullSecret is not found in any of the above two +*/}} +{{- define "k8s-triliovault-operator.imagePullSecret" -}} + {{- if eq .Values.imagePullSecret "" -}} + {{- if eq .Values.svcAccountName "" -}} + {{- printf "" -}} + {{- else -}} + {{- if (lookup "v1" "ServiceAccount" .Release.Namespace .Values.svcAccountName).imagePullSecrets -}} + {{- if (index (lookup "v1" "ServiceAccount" .Release.Namespace .Values.svcAccountName).imagePullSecrets 0).name -}} + {{- printf "%s" (index (lookup "v1" "ServiceAccount" .Release.Namespace .Values.svcAccountName).imagePullSecrets 0).name -}} + {{- else -}} + {{- printf "" -}} + {{- end -}} + {{- else -}} + {{- printf "" -}} + {{- end -}} + {{- end -}} + {{- else -}} + {{- printf "%s" .Values.imagePullSecret -}} + {{- end -}} +{{- end -}} + +{{- define "k8s-triliovault-operator.observability" -}} +app.kubernetes.io/part-of: k8s-triliovault-operator +app.kubernetes.io/managed-by: k8s-triliovault-operator +app.kubernetes.io/name: k8s-triliovault-operator +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole.yaml new file mode 100644 index 000000000..443e499d4 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole.yaml @@ -0,0 +1,148 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{template "k8s-triliovault-operator.name" .}}-{{.Release.Namespace}}-manager-role + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{template "k8s-triliovault-operator.appName" .}}-manager-role +rules: + - apiGroups: + - '*' + resources: + - '*' + verbs: + - get + - list + - watch + - apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + - customresourcedefinitions/finalizers + verbs: + - create + - update + - delete + - patch + - apiGroups: + - "" + resources: + - serviceaccounts + - services + - secrets + - events + - pods + - endpoints + - configmaps + - secrets/finalizers + - events/finalizers + - pods/finalizers + - endpoints/finalizers + - configmaps/finalizers + - services/finalizers + - serviceaccounts/finalizers + verbs: + - create + - update + - delete + - patch + - apiGroups: + - admissionregistration.k8s.io + resources: + - validatingwebhookconfigurations + - mutatingwebhookconfigurations + - validatingwebhookconfigurations/finalizers + - mutatingwebhookconfigurations/finalizers + verbs: + - create + - update + - delete + - patch + - apiGroups: + - apps + resources: + - deployments + - deployments/finalizers + verbs: + - create + - update + - delete + - patch + - apiGroups: + - rbac.authorization.k8s.io + resources: + - clusterroles + - clusterrolebindings + - roles + - rolebindings + - clusterroles/finalizers + - clusterrolebindings/finalizers + - roles/finalizers + - rolebindings/finalizers + verbs: + - create + - update + - delete + - patch + - bind + - escalate + - apiGroups: + - triliovault.trilio.io + resources: + - '*' + verbs: + - '*' + - apiGroups: + - "" + resources: + - namespaces + - namespaces/finalizers + verbs: + - update + - apiGroups: + - batch + resources: + - cronjobs + - cronjobs/finalizers + verbs: + - create + - delete + - update + - patch + - apiGroups: + - batch + resources: + - jobs + - jobs/finalizers + verbs: + - create + - delete + - apiGroups: + - policy + resources: + - poddisruptionbudgets + - poddisruptionbudgets/finalizers + verbs: + - create + - update + - patch + - delete + - apiGroups: + - networking.k8s.io + resources: + - ingresses + - ingressclasses + - ingresses/finalizers + - ingressclasses/finalizers + verbs: + - create + - patch + - update + - delete + - apiGroups: + - networking.k8s.io + resources: + - ingresses/status + verbs: + - update diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole_binding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole_binding.yaml new file mode 100644 index 000000000..e0f0bdb5f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/clusterrole_binding.yaml @@ -0,0 +1,15 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ template "k8s-triliovault-operator.name" . }}-{{ .Release.Namespace }}-manager-rolebinding + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-manager-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ template "k8s-triliovault-operator.name" . }}-{{ .Release.Namespace }}-manager-role +subjects: +- kind: ServiceAccount + name: {{ template "k8s-triliovault-operator.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/deployment.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/deployment.yaml new file mode 100644 index 000000000..85069b320 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/deployment.yaml @@ -0,0 +1,390 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "k8s-triliovault-operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }} + {{- if .Values.global.azure }} + azure-extensions-usage-release-identifier: {{ .Release.Name }} + {{- end }} +spec: + strategy: + type: RollingUpdate + rollingUpdate: + maxSurge: 25% + maxUnavailable: 25% + selector: + matchLabels: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" + replicas: {{ .Values.replicaCount }} + template: + metadata: + {{- if .Values.podAnnotations }} + annotations: + {{- range $key, $value := .Values.podAnnotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + {{- end }} + labels: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" + {{- if .Values.global.azure }} + azure-extensions-usage-release-identifier: {{ .Release.Name }} + {{- end }} + {{- include "k8s-triliovault-operator.labels" . | nindent 8 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }} + {{- range $key, $value := .Values.podLabels }} + {{ $key }}: {{ $value | quote }} + {{- end }} + spec: + hostNetwork: {{ .Values.podSpec.hostNetwork }} + hostIPC: {{ .Values.podSpec.hostIPC }} + hostPID: {{ .Values.podSpec.hostPID }} + {{- if .Values.priorityClassName }} + {{ template "k8s-triliovault-operator.priorityClassValidator" .}} + priorityClassName: {{ .Values.priorityClassName }} + {{- end }} + {{- if .Values.securityContext }} + securityContext: + {{- toYaml .Values.podSpec.securityContext | nindent 8 }} + {{- end }} + {{- if include "k8s-triliovault-operator.imagePullSecret" . }} + imagePullSecrets: + - name: {{ template "k8s-triliovault-operator.imagePullSecret" . }} + {{- end }} + containers: + - name: k8s-triliovault-operator + {{- if .Values.global.azure }} + image: {{index .Values "global" "azure" "images" "k8s-triliovault-operator" "registry" }}/{{index .Values "global" "azure" "images" "k8s-triliovault-operator" "image" }}@{{index .Values "global" "azure" "images" "k8s-triliovault-operator" "digest" }} + {{- else }} + image: {{ .Values.registry }}/{{ index .Values "k8s-triliovault-operator" "repository" }}:{{ .Values.tag }} + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy | quote }} + env: + {{- if .Values.proxySettings.PROXY_ENABLED }} + - name: HTTP_PROXY + value: {{ .Values.proxySettings.HTTP_PROXY }} + - name: HTTPS_PROXY + value: {{ .Values.proxySettings.HTTPS_PROXY }} + - name: NO_PROXY + value: {{ .Values.proxySettings.NO_PROXY }} + {{- if .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + - name: PROXY_CA_CONFIGMAP + value: {{ .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + {{- end }} + {{- end }} + - name: MASTER_ENCRYPTION_KEY_NAMESPACE + value: {{ .Values.masterEncryptionKeyConfig.namespace | default .Release.Namespace }} + - name: MASTER_ENCRYPTION_KEY_NAME + value: {{ .Values.masterEncryptionKeyConfig.name }} + {{- if .Values.observability.enabled }} + - name: OBSERVABILITY_SECRET_NAME + value: {{ .Values.observability.name }} + - name: OBSERVABILITY_SECRET_NAMESPACE + value: {{ .Values.observability.namespace | default .Release.Namespace }} + {{- end}} + - name: INSTALL_NAMESPACE + value: {{ .Release.Namespace }} + - name: REGISTRY + value: {{ .Values.registry }} + - name: RELATED_IMAGE_INGRESS_CONTROLLER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "ingress-controller" "image"}}@{{index .Values "global" "azure" "images" "ingress-controller" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "ingress-controller" "image"}}:{{ index .Values "relatedImages" "ingress-controller" "tag" }} + {{- end }} + - name: RELATED_IMAGE_KUBE_CERTGEN + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "kube-certgen" "image"}}@{{index .Values "global" "azure" "images" "kube-certgen" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "kube-certgen" "image"}}:{{ index .Values "relatedImages" "kube-certgen" "tag" }} + {{- end }} + - name: RELATED_IMAGE_METAMOVER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.metamover.image }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.metamover.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_CONTROL_PLANE + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{index .Values "relatedImages" "control-plane" "image" }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{index .Values "relatedImages" "control-plane" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_WEB + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.web.image }}@{{index .Values "global" "azure" "images" "web" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.web.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_WEB_BACKEND + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "web-backend" "image" }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "web-backend" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_EXPORTER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.exporter.image }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.exporter.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_ADMISSION_WEBHOOK + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "admission-webhook" "image" }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "admission-webhook" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_ANALYZER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.analyzer.image }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.analyzer.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_DATAMOVER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.datamover.image }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.datamover.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_DATASTORE_ATTACHER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "datastore-attacher" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "datastore-attacher" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_SCHEDULER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-scheduler" "image" }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-scheduler" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_CLEANER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-cleaner" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-cleaner" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_TARGET_BROWSER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "target-browser" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "target-browser" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_RETENTION + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-retention" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "backup-retention" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_HOOK + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.hook.image }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.hook.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_RESOURCE_CLEANER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "resource-cleaner" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "resource-cleaner" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_TVK_INIT + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "tvk-init" "image" }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "tvk-init" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_DEX + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.dex.image }}@{{index .Values "global" "azure" "images" "dex" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.dex.image }}:{{ .Values.relatedImages.dex.tag }} + {{- end }} + - name: RELATED_IMAGE_MINIO + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.minio.image }}@{{index .Values "global" "azure" "images" "control-plane" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.minio.image }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: RELATED_IMAGE_NATS + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.nats.image }}@{{index .Values "global" "azure" "images" "nats" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.nats.image }}:{{ .Values.relatedImages.nats.tag }} + {{- end }} + - name: RELATED_IMAGE_SERVICE_MANAGER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{index .Values "relatedImages" "service-manager" "image" }}@{{index .Values "global" "azure" "images" "event-stack" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{index .Values "relatedImages" "service-manager" "image" }}:{{ .Values.relatedImages.tags.event }} + {{- end }} + - name: RELATED_IMAGE_SYNCER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.syncer.image }}@{{index .Values "global" "azure" "images" "event-stack" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.syncer.image }}:{{ .Values.relatedImages.tags.event }} + {{- end }} + - name: RELATED_IMAGE_WATCHER + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.watcher.image }}@{{index .Values "global" "azure" "images" "event-stack" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ .Values.relatedImages.watcher.image }}:{{ .Values.relatedImages.tags.event }} + {{- end }} + - name: RELATED_IMAGE_CONTINUOUS_RESTORE + {{- if .Values.global.azure }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "continuous-restore" "image" }}@{{index .Values "global" "azure" "images" "datamover" "digest" }} + {{- else }} + value: {{ .Values.registry }}/{{ index .Values "relatedImages" "continuous-restore" "image" }}:{{ .Values.relatedImages.tags.tvk }} + {{- end }} + - name: ADMISSION_MUTATION_CONFIG + value: {{ template "k8s-triliovault-operator.name" . }}-mutating-webhook-configuration + - name: ADMISSION_VALIDATION_CONFIG + value: {{ template "k8s-triliovault-operator.name" . }}-validating-webhook-configuration + - name: RELEASE_VERSION + value: !!str {{ .Chart.AppVersion }} + - name: OPERATOR_INSTANCE_NAME + value: {{ template "k8s-triliovault-operator.appName" . }} + {{- if .Values.podAnnotations }} + - name: POD_ANNOTATIONS + value: {{ .Values.podAnnotations | toPrettyJson | quote }} + {{- end }} + {{- if .Values.podLabels }} + - name: POD_LABELS + value: {{ .Values.podLabels | toPrettyJson | quote }} + {{- end }} + - name: PRIORITY_CLASS_NAME + value: {{ .Values.priorityClassName }} + livenessProbe: + httpGet: + path: /healthz + port: 8081 + scheme: HTTP + initialDelaySeconds: 60 + periodSeconds: 30 + timeoutSeconds: 2 + successThreshold: 1 + failureThreshold: 3 + readinessProbe: + httpGet: + path: /readyz + port: 8081 + scheme: HTTP + initialDelaySeconds: 5 + periodSeconds: 5 + timeoutSeconds: 1 + successThreshold: 1 + failureThreshold: 3 + volumeMounts: + {{- if and .Values.proxySettings.PROXY_ENABLED .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + - name: proxy-ca-cert + mountPath: /proxy-certs + readOnly: true + {{- end }} + {{- if .Values.tls.enable }} + - name: helm-tls-certs + mountPath: /root/.helm + readOnly: true + {{- if .Values.tls.verify }} + - name: helm-tls-ca + mountPath: /root/.helm/ca.crt + readOnly: true + {{- end }} + {{- end }} + - mountPath: /tmp/k8s-webhook-server/serving-certs + name: webhook-certs + readOnly: true + {{- if .Values.securityContext }} + securityContext: + {{- toYaml .Values.securityContext | nindent 12 }} + {{- end }} + resources: + limits: + cpu: 200m + memory: 512Mi + requests: + cpu: 10m + memory: 256Mi + initContainers: + - name: webhook-init + {{- if .Values.global.azure }} + image: {{index .Values "global" "azure" "images" "operator-webhook-init" "registry" }}/{{index .Values "global" "azure" "images" "operator-webhook-init" "image" }}@{{index .Values.global.azure "images" "operator-webhook-init" "digest" }} + {{- else }} + image: {{ .Values.registry }}/{{ index .Values "operator-webhook-init" "repository" }}:{{ .Values.tag }} + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy | quote }} + {{- if .Values.securityContext }} + securityContext: + {{- toYaml .Values.securityContext | nindent 12 }} + {{- end }} + env: + {{- if .Values.proxySettings.PROXY_ENABLED }} + - name: HTTP_PROXY + value: {{ .Values.proxySettings.HTTP_PROXY }} + - name: HTTPS_PROXY + value: {{ .Values.proxySettings.HTTPS_PROXY }} + - name: NO_PROXY + value: {{ .Values.proxySettings.NO_PROXY }} + {{- if .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + - name: PROXY_CA_CONFIGMAP + value: {{ .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + {{- end }} + {{- end }} + - name: MASTER_ENCRYPTION_KEY_NAMESPACE + value: {{ .Values.masterEncryptionKeyConfig.namespace | default .Release.Namespace }} + - name: MASTER_ENCRYPTION_KEY_NAME + value: {{ .Values.masterEncryptionKeyConfig.name }} + - name: RELEASE_VERSION + value: !!str {{ .Chart.AppVersion }} + - name: ADMISSION_MUTATION_CONFIG + value: {{ template "k8s-triliovault-operator.name" . }}-mutating-webhook-configuration + - name: ADMISSION_VALIDATION_CONFIG + value: {{ template "k8s-triliovault-operator.name" . }}-validating-webhook-configuration + - name: NAMESPACE_VALIDATION_CONFIG + value: {{ template "k8s-triliovault-operator.name" . }}-ns-validating-webhook-configuration + - name: WEBHOOK_SERVICE + value: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + - name: WEBHOOK_NAMESPACE + value: {{ .Release.Namespace }} + - name: SECRET_NAME + value: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-certs + {{- if and .Values.proxySettings.PROXY_ENABLED .Values.proxySettings.CA_BUNDLE_CONFIGMAP }} + volumeMounts: + - name: proxy-ca-cert + mountPath: /proxy-certs + readOnly: true + {{- end }} + serviceAccountName: {{ template "k8s-triliovault-operator.serviceAccountName" . }} + {{- if .Values.nodeSelector }} + nodeSelector: {{- .Values.nodeSelector | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.affinity }} + affinity: + {{- toYaml .Values.affinity | nindent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- end }} + volumes: + {{- if .Values.tls.enable }} + - name: helm-tls-certs + secret: + secretName: {{ .Values.tls.secretName }} + defaultMode: 0400 + {{- if .Values.tls.verify }} + - name: helm-tls-ca + configMap: + name: {{ template "k8s-triliovault-operator.fullname" . }}-helm-tls-ca-config + defaultMode: 0600 + {{- end }} + {{- end }} + - name: webhook-certs + secret: + defaultMode: 420 + secretName: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-certs diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/mutating-webhook.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/mutating-webhook.yaml new file mode 100644 index 000000000..dc7902c62 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/mutating-webhook.yaml @@ -0,0 +1,28 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ template "k8s-triliovault-operator.name" . }}-mutating-webhook-configuration + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-mutating-webhook-configuration +webhooks: +- clientConfig: + service: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + path: /mutate-triliovault-trilio-io-v1-triliovaultmanager + failurePolicy: Fail + name: v1-tvm-mutation.trilio.io + rules: + - apiGroups: + - triliovault.trilio.io + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - triliovaultmanagers + sideEffects: None + admissionReviewVersions: + - v1 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/preflight_job_preinstall_hook.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/preflight_job_preinstall_hook.yaml new file mode 100644 index 000000000..7f2b367c4 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/preflight_job_preinstall_hook.yaml @@ -0,0 +1,203 @@ +{{- if .Values.preflight.enabled -}} +{{- template "k8s-triliovault-operator.preFlightValidation" . }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{template "k8s-triliovault-operator.name" .}}-{{.Release.Namespace}}-preflight-role + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{template "k8s-triliovault-operator.appName" .}}-preflight-role + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": hook-failed, hook-succeeded + "helm.sh/hook-weight": "1" +rules: + - apiGroups: + - '*' + resources: + - '*' + verbs: + - get + - list + - apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - create + - update + - delete + - patch + - apiGroups: + - "" + resources: + - serviceaccounts + - pods + - persistentvolumeclaims + - pods/exec + verbs: + - create + - update + - delete + - patch + - apiGroups: + - rbac.authorization.k8s.io + resources: + - clusterroles + - clusterrolebindings + verbs: + - create + - update + - delete + - patch + - apiGroups: + - "" + resources: + - namespaces + verbs: + - get + - list + - apiGroups: + - batch + resources: + - jobs + verbs: + - create + - delete + - apiGroups: + - storage.k8s.io + resources: + - storageclasses + verbs: + - get + - list + - apiGroups: + - snapshot.storage.k8s.io + resources: + - volumesnapshots + - volumesnapshotclasses + verbs: + - get + - list + - create + - update + - delete + - patch + +--- +{{- if eq .Values.svcAccountName "" }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ template "k8s-triliovault-operator.preflightServiceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-preflight-service-account + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": hook-failed, hook-succeeded + "helm.sh/hook-weight": "2" +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ template "k8s-triliovault-operator.name" . }}-{{ .Release.Namespace }}-preflight-rolebinding + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-preflight-rolebinding + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": hook-failed, hook-succeeded + "helm.sh/hook-weight": "3" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ template "k8s-triliovault-operator.name" . }}-{{ .Release.Namespace }}-preflight-role +subjects: + - kind: ServiceAccount + name: {{ template "k8s-triliovault-operator.preflightServiceAccountName" . }} + namespace: {{ .Release.Namespace }} + +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "k8s-triliovault-operator.name" . }}-preflight-job-preinstall-hook-{{ randAlphaNum 4 | lower }} + namespace: {{ .Release.Namespace }} + labels: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-preflight-job-preinstall-hook + {{- if .Values.global.azure }} + azure-extensions-usage-release-identifier: {{ .Release.Name }} + {{- end }} + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": hook-succeeded + "helm.sh/hook-weight": "4" +spec: + backoffLimit: 0 + ttlSecondsAfterFinished: 3600 + template: + spec: + containers: + - name: preflight + {{- if .Values.global.azure }} + image: {{ index .Values "registry" }}/{{ index .Values "preflight" "repository" }}@{{index .Values "global" "azure" "images" "preflight" "digest" }} + {{- else }} + image: {{ index .Values "registry" }}/{{ index .Values "preflight" "repository" }}:{{ index .Values "preflight" "imageTag" }} + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy | quote }} + command: + - /bin/sh + - -c + - >- + /opt/tvk-plugins/preflight run --in-cluster + --log-level={{ .Values.preflight.logLevel }} + --namespace={{ .Release.Namespace }} + {{- if .Values.preflight.cleanupOnFailure }} + --cleanup-on-failure + {{- end }} + {{- if .Values.preflight.imagePullSecret }} + --image-pull-secret={{ .Values.preflight.imagePullSecret }} + {{- end }} + {{- if .Values.preflight.limits }} + --limits={{ .Values.preflight.limits }} + {{- end }} + {{- if .Values.preflight.localRegistry }} + --local-registry={{ .Values.preflight.localRegistry }} + {{- end }} + {{- if .Values.preflight.nodeSelector }} + --node-selector={{ .Values.preflight.nodeSelector }} + {{- end }} + {{- if .Values.preflight.pvcStorageRequest }} + --pvc-storage-request={{ .Values.preflight.pvcStorageRequest }} + {{- end }} + {{- if .Values.preflight.requests }} + --requests={{ .Values.preflight.requests }} + {{- end }} + {{- if .Values.preflight.storageClass }} + --storage-class={{ .Values.preflight.storageClass }} + {{- end }} + {{- if .Values.preflight.volumeSnapshotClass }} + --volume-snapshot-class={{ .Values.preflight.volumeSnapshotClass }} + {{- end }} + {{- if .Values.nodeSelector }} + nodeSelector: {{- .Values.nodeSelector | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.affinity }} + affinity: + {{- toYaml .Values.affinity | nindent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- end }} + restartPolicy: Never + terminationGracePeriodSeconds: 0 + serviceAccountName: {{ template "k8s-triliovault-operator.preflightServiceAccountName" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/secret.yaml new file mode 100644 index 000000000..22f56e848 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/secret.yaml @@ -0,0 +1,9 @@ +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-certs + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-webhook-certs +type: Opaque diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/serviceAccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/serviceAccount.yaml new file mode 100644 index 000000000..c36e39bd0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/serviceAccount.yaml @@ -0,0 +1,14 @@ +{{- if eq .Values.svcAccountName "" }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ template "k8s-triliovault-operator.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-service-account +{{- if .Values.imagePullSecret }} +imagePullSecrets: +- name: {{ .Values.imagePullSecret }} +{{- end}} +{{- end }} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/validating-webhook.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/validating-webhook.yaml new file mode 100644 index 000000000..66d1044d6 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/validating-webhook.yaml @@ -0,0 +1,104 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: {{ template "k8s-triliovault-operator.name" . }}-validating-webhook-configuration + labels: + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-validating-webhook-configuration +webhooks: +- clientConfig: + service: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + path: /validate-triliovault-trilio-io-v1-triliovaultmanager + failurePolicy: Fail + name: v1-tvm-validation.trilio.io + rules: + - apiGroups: + - triliovault.trilio.io + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - triliovaultmanagers + sideEffects: None + admissionReviewVersions: + - v1 +- clientConfig: + service: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + path: /validate-core-v1-secret + failurePolicy: Fail + name: v1-encryption-secret-validation.trilio.io + objectSelector: + matchExpressions: + - key: triliovault.trilio.io/master-secret + operator: Exists + rules: + - apiGroups: + - "*" + apiVersions: + - v1 + operations: + - DELETE + - UPDATE + resources: + - secrets + sideEffects: None + admissionReviewVersions: + - v1 +- clientConfig: + service: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + path: /validate-core-v1-namespace + failurePolicy: Fail + name: v1-tvm-ns-validation.trilio.io + namespaceSelector: + matchExpressions: + - key: trilio-operator-label + operator: In + values: + - {{ .Release.Namespace }} + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - DELETE + resources: + - namespaces + scope: '*' + sideEffects: None + admissionReviewVersions: + - v1 +{{- if .Values.observability.enabled }} +- clientConfig: + service: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + path: /validate-core-v1-secret + failurePolicy: Ignore + name: v1-observability-secret-validation.trilio.io + objectSelector: + matchExpressions: + - key: triliovault.trilio.io/observability + operator: Exists + rules: + - apiGroups: + - "*" + apiVersions: + - v1 + operations: + - DELETE + - UPDATE + resources: + - secrets + sideEffects: None + admissionReviewVersions: + - v1 +{{- end }} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/templates/webhook-service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/webhook-service.yaml new file mode 100644 index 000000000..8717867d7 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/templates/webhook-service.yaml @@ -0,0 +1,17 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "k8s-triliovault-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + labels: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" + {{- include "k8s-triliovault-operator.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "k8s-triliovault-operator.appName" . }}-webhook-service +spec: + ports: + - port: 443 + targetPort: 9443 + selector: + app: {{ template "k8s-triliovault-operator.fullname" . }} + release: "{{ .Release.Name }}" diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/values.yaml new file mode 100644 index 000000000..836f0d06a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/values.yaml @@ -0,0 +1,247 @@ +## TrilioVault Operator +global: + urlPath: "/" +registry: "quay.io/triliodata" +operator-webhook-init: + repository: operator-webhook-init +k8s-triliovault-operator: + repository: k8s-triliovault-operator +tag: "5.0.0" +# create image pull secrets and specify the name here. +imagePullSecret: "" +priorityClassName: "" +preflight: + enabled: false + repository: preflight + imageTag: "1.3.1" + logLevel: "INFO" + cleanupOnFailure: false + imagePullSecret: "" + limits: "" + localRegistry: "" + nodeSelector: "" + pvcStorageRequest: "" + requests: "" + storageClass: "" + volumeSnapshotClass: "" +# Affinity rules for scheduling the Pod of this application. +# https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#node-affinity +affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: kubernetes.io/arch + operator: In + values: + - amd64 + - ppc64le +# Node selection constraints for scheduling Pods of this application. +# https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector +nodeSelector: {} +# Taints to be tolerated by Pods of this application. +# https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +tolerations: [] +masterEncryptionKeyConfig: + name: "triliovault-master-encryption-key" + namespace: "" +image: + pullPolicy: Always +tls: + secretName: "helm-client-certs" + verify: false + enable: false + keyFile: "tls.key" + certFile: "tls.crt" + caContent: "" + hostname: "" +nameOverride: "" +replicaCount: 1 +proxySettings: + PROXY_ENABLED: false + NO_PROXY: "" + HTTP_PROXY: "" + HTTPS_PROXY: "" + CA_BUNDLE_CONFIGMAP: "" +podSpec: + hostIPC: false + hostNetwork: false + hostPID: false + securityContext: + runAsNonRoot: true + runAsUser: 1001 +securityContext: + allowPrivilegeEscalation: false + privileged: false + readOnlyRootFilesystem: false + runAsNonRoot: true + runAsUser: 1001 + capabilities: + drop: + - ALL +installTVK: + enabled: true + applicationScope: Cluster + tvkInstanceName: "" + ingressConfig: + host: "" + tlsSecretName: "" + annotations: {} + ingressClass: "" + ComponentConfiguration: + ingressController: + enabled: true + service: + type: NodePort +observability: + enabled: false + name: "tvk-integration" + logging: + loki: + enabled: true + fullnameOverride: "loki" + persistence: + enabled: true + accessModes: + - ReadWriteOnce + size: 10Gi + config: + limits_config: + reject_old_samples_max_age: 168h + table_manager: + retention_period: 168h + image: + registry: docker.io + promtail: + enabled: true + fullnameOverride: "promtail" + config: + clients: + - url: http://loki:3100/loki/api/v1/push + image: + registry: docker.io + monitoring: + prometheus: + enabled: true + fullnameOverride: "prom" + server: + enabled: true + fullnameOverride: "prom-server" + persistentVolume: + enabled: false + image: + registry: quay.io + kubeStateMetrics: + enabled: false + image: + registry: k8s.gcr.io + nodeExporter: + enabled: false + image: + registry: quay.io + pushgateway: + enabled: false + image: + registry: docker.io + alertmanager: + enabled: false + image: + registry: quay.io + configmapReload: + prometheus: + image: + registry: docker.io + alertmanager: + image: + registry: docker.io + visualization: + grafana: + grafana.ini: + server: + root_url: "%(protocol)s://%(domain)s:%(http_port)s{{- if ne .Values.global.urlPath \"/\" }}{{.Values.global.urlPath}}{{- end }}/grafana/" + enabled: true + adminPassword: "admin123" + fullnameOverride: "grafana" + service: + type: ClusterIP + image: + registry: docker.io + testFramework: + registry: docker.io + imageRenderer: + image: + registry: docker.io + sidecar: + image: + registry: quay.io + initChownData: + image: + registry: docker.io + downloadDashboardsImage: + registry: docker.io +# these annotations will be added to all tvk pods +podAnnotations: + sidecar.istio.io/inject: false +# these labels will be added to all tvk pods +podLabels: + sidecar.portshift.io/inject: false + linkerd.io/inject: disabled +relatedImages: + tags: + tvk: "5.0.0" + event: "5.0.0" + control-plane: + image: "control-plane" + metamover: + image: "datamover" + datamover: + image: "datamover" + datastore-attacher: + image: "datamover" + admission-webhook: + image: "control-plane" + analyzer: + image: "control-plane" + ingress-controller: + image: "ingress-controller" + tag: "v1.10.1" + kube-certgen: + image: "kube-certgen" + tag: "v1.4.1" + exporter: + image: "control-plane" + web: + image: "web" + web-backend: + image: "control-plane" + backup-scheduler: + image: "control-plane" + backup-cleaner: + image: "datamover" + target-browser: + image: "datamover" + backup-retention: + image: "datamover" + hook: + image: "control-plane" + resource-cleaner: + image: "datamover" + tvk-init: + image: "control-plane" + dex: + image: "dex" + tag: "2.30.7" + minio: + image: "control-plane" + nats: + image: "nats" + tag: "2.8.5" + service-manager: + image: "event-stack" + syncer: + image: "event-stack" + watcher: + image: "event-stack" + continuous-restore: + image: "datamover" +svcAccountName: "" diff --git a/index.yaml b/index.yaml index 11a27801c..86b113605 100644 --- a/index.yaml +++ b/index.yaml @@ -21625,6 +21625,34 @@ entries: - assets/jenkins/jenkins-4.3.27.tgz version: 4.3.27 k8s-triliovault-operator: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: TrilioVault for Kubernetes Operator + catalog.cattle.io/kube-version: '>=1.19.0-0' + catalog.cattle.io/release-name: k8s-triliovault-operator + apiVersion: v2 + appVersion: 5.0.0 + created: "2024-11-16T00:01:59.315332335Z" + dependencies: + - condition: observability.enabled + name: observability + repository: file://charts/observability + version: ^0.1.0 + description: K8s-TrilioVault-Operator is an operator designed to manage the K8s-TrilioVault + Application Lifecycle. + digest: 837a3a2119ee5c8b27200b39a0fa1f0f5060da0fd2f722b841ab07b7d6d0543e + home: https://github.com/trilioData/k8s-triliovault-operator + icon: file://assets/icons/k8s-triliovault-operator.png + kubeVersion: '>=1.19.0-0' + maintainers: + - email: prafull.ladha@trilio.io + name: prafull11 + name: k8s-triliovault-operator + sources: + - https://github.com/trilioData/k8s-triliovault-operator + urls: + - assets/trilio/k8s-triliovault-operator-5.0.0.tgz + version: 5.0.0 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: TrilioVault for Kubernetes Operator @@ -21878,6 +21906,35 @@ entries: - assets/trilio/k8s-triliovault-operator-3.1.1.tgz version: 3.1.1 k10: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: K10 + catalog.cattle.io/kube-version: '>= 1.17.0-0' + catalog.cattle.io/release-name: k10 + apiVersion: v2 + appVersion: 7.0.14 + created: "2024-11-16T00:01:56.510575719Z" + dependencies: + - condition: grafana.enabled + name: grafana + repository: "" + version: 8.5.8 + - condition: prometheus.server.enabled + name: prometheus + repository: "" + version: 25.28.0 + description: Kasten’s K10 Data Management Platform + digest: ef53b6554e34aa4050ef132f6785147d7f954c227561080c8763fb24cd6c40b4 + home: https://kasten.io/ + icon: file://assets/icons/k10.png + kubeVersion: '>= 1.17.0-0' + maintainers: + - email: contact@kasten.io + name: kastenIO + name: k10 + urls: + - assets/kasten/k10-7.0.1401.tgz + version: 7.0.1401 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: K10 @@ -24730,6 +24787,36 @@ entries: - assets/avesha/kubeslice-worker-1.1.1.tgz version: 1.1.1 kuma: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Kuma + catalog.cattle.io/namespace: kuma-system + catalog.cattle.io/release-name: kuma + apiVersion: v2 + appVersion: 2.9.1 + created: "2024-11-16T00:01:57.181919052Z" + description: A Helm chart for the Kuma Control Plane + digest: 08d9aaa9449221b3b8e5acb715f2e9c171923f2dff1a5bd0d58980228e70332b + home: https://github.com/kumahq/kuma + icon: file://assets/icons/kuma.svg + keywords: + - service mesh + - control plane + maintainers: + - email: jakub.dyszkiewicz@konghq.com + name: Jakub Dyszkiewicz + url: https://github.com/jakubdyszkiewicz + - email: charly.molter@konghq.com + name: Charly Molter + url: https://github.com/lahabana + - email: michael.beaumont@konghq.com + name: Mike Beaumont + url: https://github.com/michaelbeaumont + name: kuma + type: application + urls: + - assets/kuma/kuma-2.9.1.tgz + version: 2.9.1 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: Kuma @@ -28752,6 +28839,54 @@ entries: - assets/loft/loft-3.2.0.tgz version: 3.2.0 microgateway: + - annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.4/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: "" + charts.openshift.io/name: Airlock Microgateway + apiVersion: v2 + appVersion: 4.4.1 + created: "2024-11-16T00:01:53.700514743Z" + description: A Helm chart for deploying the Airlock Microgateway + digest: 2867a07cdff0938923432197b2a9c87201ff82b0d83ccbb88b738b577a6a954f + home: https://www.airlock.com/en/microgateway + icon: file://assets/icons/microgateway.svg + keywords: + - WAF + - Web Application Firewall + - WAAP + - Web Application and API protection + - OWASP + - Airlock + - Microgateway + - Security + - Filtering + - DevSecOps + - shift left + - control plane + - Operator + kubeVersion: '>=1.25.0-0' + maintainers: + - email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ + name: microgateway + sources: + - https://github.com/airlock/microgateway + type: application + urls: + - assets/airlock/microgateway-4.4.1.tgz + version: 4.4.1 - annotations: artifacthub.io/category: security artifacthub.io/license: MIT @@ -29089,6 +29224,53 @@ entries: - assets/airlock/microgateway-4.2.3.tgz version: 4.2.3 microgateway-cni: + - annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.4/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway CNI + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: "" + charts.openshift.io/name: Airlock Microgateway CNI + apiVersion: v2 + appVersion: 4.4.1 + created: "2024-11-16T00:01:53.705500386Z" + description: A Helm chart for deploying the Airlock Microgateway CNI plugin + digest: e1113ae0e35e1c9fd857e19a5ca65e6d9dc3100ac3c569113dd7da9eaf0a7dd5 + home: https://www.airlock.com/en/microgateway + icon: file://assets/icons/microgateway-cni.svg + keywords: + - WAF + - Web Application Firewall + - WAAP + - Web Application and API protection + - OWASP + - Airlock + - Microgateway + - Security + - Filtering + - DevSecOps + - shift left + - CNI + kubeVersion: '>=1.25.0-0' + maintainers: + - email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ + name: microgateway-cni + sources: + - https://github.com/airlock/microgateway + type: application + urls: + - assets/airlock/microgateway-cni-4.4.1.tgz + version: 4.4.1 - annotations: artifacthub.io/category: security artifacthub.io/license: MIT @@ -30497,6 +30679,31 @@ entries: - assets/nats/nats-0.19.15.tgz version: 0.19.15 netscaler-cpx-with-ingress-controller: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: NetScaler CPX with Ingress Controller + catalog.cattle.io/kube-version: '>=v1.16.0-0' + catalog.cattle.io/release-name: netscaler-cpx-with-ingress-controller + apiVersion: v2 + appVersion: 2.2.10 + created: "2024-11-16T00:01:57.315471785Z" + description: A Helm chart for NetScaler CPX with NetScaler ingress Controller + running as sidecar. + digest: 1e0ea2a4d05c1d1819568bbc78becabd6d4d4feab6fbf1938e295367c2ea9cdd + home: https://www.netscaler.com + icon: file://assets/icons/netscaler-cpx-with-ingress-controller.png + kubeVersion: '>=v1.16.0-0' + maintainers: + - email: priyanka.sharma@cloud.com + name: priyankash-citrix + - email: subash.dangol@cloud.com + name: subashd + name: netscaler-cpx-with-ingress-controller + sources: + - https://github.com/netscaler/netscaler-k8s-ingress-controller + urls: + - assets/netscaler/netscaler-cpx-with-ingress-controller-2.2.10.tgz + version: 2.2.10 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: NetScaler CPX with Ingress Controller @@ -30523,6 +30730,30 @@ entries: - assets/netscaler/netscaler-cpx-with-ingress-controller-2.1.4.tgz version: 2.1.4 netscaler-ingress-controller: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: NetScaler Ingress Controller + catalog.cattle.io/kube-version: '>=v1.16.0-0' + catalog.cattle.io/release-name: netscaler-ingress-controller + apiVersion: v2 + appVersion: 2.2.10 + created: "2024-11-16T00:01:57.318087902Z" + description: A Helm chart for NetScaler Ingress Controller configuring MPX/VPX. + digest: e5ec26bc4e26e780bcfa0ac3c58862c4b71d6e096cced9514c8e68b83587053b + home: https://www.netscaler.com + icon: file://assets/icons/netscaler-ingress-controller.png + kubeVersion: '>=v1.16.0-0' + maintainers: + - email: priyanka.sharma@cloud.com + name: priyankash-citrix + - email: subash.dangol@cloud.com + name: subashd + name: netscaler-ingress-controller + sources: + - https://github.com/netscaler/netscaler-k8s-ingress-controller + urls: + - assets/netscaler/netscaler-ingress-controller-2.2.10.tgz + version: 2.2.10 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: NetScaler Ingress Controller @@ -48342,4 +48573,4 @@ entries: urls: - assets/netfoundry/ziti-host-1.5.1.tgz version: 1.5.1 -generated: "2024-11-15T00:01:45.043320849Z" +generated: "2024-11-16T00:01:53.678822476Z"

    @ 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 I1^{di0LW9Cx&QzG literal 0 HcmV?d00001 diff --git a/charts/airlock/microgateway-cni/4.4.1/.helmignore b/charts/airlock/microgateway-cni/4.4.1/.helmignore new file mode 100644 index 000000000..8561d2892 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/.helmignore @@ -0,0 +1,27 @@ +# 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 +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +# Helm unit tests +/tests +/validation diff --git a/charts/airlock/microgateway-cni/4.4.1/Chart.yaml b/charts/airlock/microgateway-cni/4.4.1/Chart.yaml new file mode 100644 index 000000000..4abf51221 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/Chart.yaml @@ -0,0 +1,43 @@ +annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.4/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway CNI + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: "" + charts.openshift.io/name: Airlock Microgateway CNI +apiVersion: v2 +appVersion: 4.4.1 +description: A Helm chart for deploying the Airlock Microgateway CNI plugin +home: https://www.airlock.com/en/microgateway +icon: file://assets/icons/microgateway-cni.svg +keywords: +- WAF +- Web Application Firewall +- WAAP +- Web Application and API protection +- OWASP +- Airlock +- Microgateway +- Security +- Filtering +- DevSecOps +- shift left +- CNI +kubeVersion: '>=1.25.0-0' +maintainers: +- email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ +name: microgateway-cni +sources: +- https://github.com/airlock/microgateway +type: application +version: 4.4.1 diff --git a/charts/airlock/microgateway-cni/4.4.1/README.md b/charts/airlock/microgateway-cni/4.4.1/README.md new file mode 100644 index 000000000..77c0a31b4 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/README.md @@ -0,0 +1,137 @@ +# Airlock Microgateway CNI + +![Version: 4.4.1](https://img.shields.io/badge/Version-4.4.1-informational?style=flat-square) ![AppVersion: 4.4.1](https://img.shields.io/badge/AppVersion-4.4.1-informational?style=flat-square) + +*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* + + + + + Microgateway + + +Modern application security is embedded in the development workflow and follows DevSecOps paradigms. Airlock Microgateway is the perfect fit for these requirements. It is a lightweight alternative to the Airlock Gateway appliance, optimized for Kubernetes environments. Airlock Microgateway protects your applications and microservices with the tried-and-tested Airlock security features against attacks, while also providing a high degree of scalability. +__This Helm chart is part of Airlock Microgateway. See our [GitHub repo](https://github.com/airlock/microgateway/tree/4.4.1).__ + +### Features +* Kubernetes native integration with sidecar injection and Gateway API support +* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction +* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication +* Content security filters for protecting against known attacks (OWASP Top 10) +* Access control using OpenID Connect to allow only authenticated users to access the protected services +* API security features like JSON parsing, OpenAPI specification enforcement or GraphQL schema validation + +For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. + +## Documentation and links + +Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. + +* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) +* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) +* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) +* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) +* [GitHub](https://github.com/airlock/microgateway) + +# Quick start guide + +The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. + +## Prerequisites +* [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) + +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). + ```bash + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' -f https://raw.githubusercontent.com/airlock/microgateway/4.4.1/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' -f https://raw.githubusercontent.com/airlock/microgateway/4.4.1/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + > **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). + +2. (Recommended) You can verify the correctness of the installation with `helm test`. + ```bash + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.4.1' + ``` + + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. + +## Support + +### Premium support +If you have a paid license, please follow the [premium support process](https://techzone.ergon.ch/support-process). + +### Community support +For the community edition, check our **[Airlock community forum](https://forum.airlock.com/)** for FAQs or register to post your question. +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | +| commonAnnotations | object | `{}` | Annotations to add to all resources. | +| commonLabels | object | `{}` | Labels to add to all resources. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | +| fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:fa2f5d8587024f0d0b29505204c964002cfd7facf79748ccc98b8caf1a70f0d8"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.4.1"` | Image tag to pull. | +| imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | +| tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | + +## License +View the [detailed license terms](https://www.airlock.com/en/airlock-license) for the software contained in this image. +* Decompiling or reverse engineering is not permitted. +* Using any of the deny rules or parts of these filter patterns outside of the image is not permitted. + +Airlock® is a security innovation by [ergon](https://www.ergon.ch/en) + + + + + + + Airlock Secure Access Hub + + diff --git a/charts/airlock/microgateway-cni/4.4.1/gke-values.yaml b/charts/airlock/microgateway-cni/4.4.1/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway-cni/4.4.1/openshift-values.yaml b/charts/airlock/microgateway-cni/4.4.1/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway-cni/4.4.1/questions.yml b/charts/airlock/microgateway-cni/4.4.1/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/NOTES.txt b/charts/airlock/microgateway-cni/4.4.1/templates/NOTES.txt new file mode 100644 index 000000000..bb94ff521 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/NOTES.txt @@ -0,0 +1,15 @@ +Thank you for installing Airlock Microgateway CNI. + +Please ensure that the helm values'.config.cniNetDir' and '.config.cniBinDir' are configured for your Kubernetes distribution. +For further information, consider our manual https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }}. +The chapter 'Setup > Installation' describes how to set those settings correctly. + +Further information: +* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }} +* Airlock Microgateway Labs: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=helm + +Next steps: +* Install Airlock Microgateway (if not done already) + https://artifacthub.io/packages/helm/airlock-microgateway/microgateway + +Your release version is {{ .Chart.Version }}. \ No newline at end of file diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/_helpers.tpl b/charts/airlock/microgateway-cni/4.4.1/templates/_helpers.tpl new file mode 100644 index 000000000..996491a87 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/_helpers.tpl @@ -0,0 +1,101 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Convert an image configuration object into an image ref string. +*/}} +{{- define "airlock-microgateway-cni.image" -}} + {{- if .digest -}} + {{- printf "%s@%s" .repository .digest -}} + {{- else if .tag -}} + {{- printf "%s:%s" .repository .tag -}} + {{- else -}} + {{- printf "%s" .repository -}} + {{- end -}} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. +If release name contains chart name it will be used as a full name. +*/}} +{{- define "airlock-microgateway-cni.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "airlock-microgateway-cni.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- with .Values.commonLabels }} +{{ toYaml .}} +{{- end }} +{{- end }} + +{{/* +Common labels without component +*/}} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} +{{- end }} + +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{- define "airlock-microgateway-cni.isSemver" -}} +{{- regexMatch `^(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-]+)*))?$` . -}} +{{- end -}} + +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} + {{- $version := (semver .Chart.AppVersion) -}} + {{- $version.Major }}.{{ $version.Minor -}} +{{- else -}} + {{- print "latest" -}} +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/clusterrole.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/clusterrolebinding.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/configmap.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/daemonset.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/network-attachment-definition.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/scc-role.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/scc-rolebinding.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/serviceaccount.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/tests/rbac.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/tests/rbac.yaml new file mode 100644 index 000000000..744799333 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/tests/rbac.yaml @@ -0,0 +1,64 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" +subjects: +- kind: ServiceAccount + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +rules: +- apiGroups: + - "apps" + resources: + - daemonsets + resourceNames: + - {{ include "airlock-microgateway-cni.fullname" . }} + verbs: + - get + - watch + - list +- apiGroups: + - "" + resources: + - pods + - pods/log + verbs: + - get + - list +{{- if .Values.rbac.createSCCRole }} +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/templates/tests/test-install.yaml b/charts/airlock/microgateway-cni/4.4.1/templates/tests/test-install.yaml new file mode 100644 index 000000000..12d8c8de7 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/templates/tests/test-install.yaml @@ -0,0 +1,103 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: test-install + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + restartPolicy: Never + containers: + - name: test + image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true + command: + - sh + - -c + - | + set -eu + + fail() { + echo "Error: ${1}" + echo "" + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer + exit 1 + } + + containsMGWCNIConf() { + cat "${1}" | grep -qe '"type":.*"{{ include "airlock-microgateway-cni.fullname" . }}"' + } + + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' + fi + + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' + fi + + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' + fi + + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac + + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.4.1/values.schema.json b/charts/airlock/microgateway-cni/4.4.1/values.schema.json new file mode 100644 index 000000000..e087bd700 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/values.schema.json @@ -0,0 +1,225 @@ +{ + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "nameOverride": { + "type": "string" + }, + "fullnameOverride": { + "type": "string" + }, + "commonLabels": { + "$ref": "#/definitions/StringMap" + }, + "commonAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "imagePullSecrets": { + "type": "array", + "items": { + "type": "object", + "properties": { + "name": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "name" + ], + "additionalProperties": true + } + }, + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "createSCCRole": { + "type": "boolean" + } + }, + "required": [ + "create", + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" + ], + "additionalProperties": false + }, + "tests": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "additionalProperties": false + }, + "global": { + "type": "object" + } + }, + "required": [ + "affinity", + "commonAnnotations", + "commonLabels", + "config", + "fullnameOverride", + "image", + "imagePullSecrets", + "multusNetworkAttachmentDefinition", + "nameOverride", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", + "tests" + ], + "additionalProperties": false, + "definitions": { + "StringMap": { + "type": "object", + "additionalProperties": { + "type": "string" + } + }, + "Image": { + "type": "object", + "properties": { + "repository": { + "type": "string", + "minLength": 1 + }, + "tag": { + "type": "string" + }, + "digest": { + "type": "string", + "pattern": "^$|^sha256:[a-f0-9]{64}$" + }, + "pullPolicy": { + "type": "string", + "enum": [ + "Always", + "IfNotPresent", + "Never" + ] + } + }, + "required": [ + "digest", + "pullPolicy", + "repository", + "tag" + ], + "additionalProperties": false + } + } +} diff --git a/charts/airlock/microgateway-cni/4.4.1/values.yaml b/charts/airlock/microgateway-cni/4.4.1/values.yaml new file mode 100644 index 000000000..d1116802d --- /dev/null +++ b/charts/airlock/microgateway-cni/4.4.1/values.yaml @@ -0,0 +1,85 @@ +# -- Allows overriding the name to use instead of "microgateway-cni". +nameOverride: "" +# -- Allows overriding the name to use as full name of resources. +fullnameOverride: "" +# -- Labels to add to all resources. +commonLabels: {} +# -- Annotations to add to all resources. +commonAnnotations: {} +# -- ImagePullSecrets to use when pulling images. +imagePullSecrets: [] +# - name: myRegistryKeySecretName + +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.4.1" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:fa2f5d8587024f0d0b29505204c964002cfd7facf79748ccc98b8caf1a70f0d8" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. + create: false + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system +tests: + # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). + # If set to false, `helm test` will not run any tests. + enabled: false diff --git a/charts/airlock/microgateway/4.2.3/.helmignore b/charts/airlock/microgateway/4.2.3/.helmignore index 101ff5ac5..8561d2892 100644 --- a/charts/airlock/microgateway/4.2.3/.helmignore +++ b/charts/airlock/microgateway/4.2.3/.helmignore @@ -21,8 +21,7 @@ .idea/ *.tmproj .vscode/ -# CRDs kustomization.yaml -/crds/kustomization.yaml + # Helm unit tests /tests /validation diff --git a/charts/airlock/microgateway/4.2.3/Chart.yaml b/charts/airlock/microgateway/4.2.3/Chart.yaml index a248008ae..f4d50752c 100644 --- a/charts/airlock/microgateway/4.2.3/Chart.yaml +++ b/charts/airlock/microgateway/4.2.3/Chart.yaml @@ -9,15 +9,15 @@ annotations: - name: Airlock Microgateway Forum url: https://forum.airlock.com/ catalog.cattle.io/certified: partner - catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/display-name: Airlock Microgateway CNI catalog.cattle.io/kube-version: '>=1.25.0-0' - catalog.cattle.io/release-name: microgateway - charts.openshift.io/name: Airlock Microgateway + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI apiVersion: v2 appVersion: 4.2.3 -description: A Helm chart for deploying the Airlock Microgateway +description: A Helm chart for deploying the Airlock Microgateway CNI plugin home: https://www.airlock.com/en/microgateway -icon: file://assets/icons/microgateway.svg +icon: file://assets/icons/microgateway-cni.svg keywords: - WAF - Web Application Firewall @@ -30,14 +30,13 @@ keywords: - Filtering - DevSecOps - shift left -- control plane -- Operator +- CNI kubeVersion: '>=1.25.0-0' maintainers: - email: support@airlock.com name: Airlock url: https://www.airlock.com/ -name: microgateway +name: microgateway-cni sources: - https://github.com/airlock/microgateway type: application diff --git a/charts/airlock/microgateway/4.2.3/README.md b/charts/airlock/microgateway/4.2.3/README.md index 1a75b9b2c..2c5823d06 100644 --- a/charts/airlock/microgateway/4.2.3/README.md +++ b/charts/airlock/microgateway/4.2.3/README.md @@ -1,4 +1,4 @@ -# Airlock Microgateway +# Airlock Microgateway CNI ![Version: 4.2.3](https://img.shields.io/badge/Version-4.2.3-informational?style=flat-square) ![AppVersion: 4.2.3](https://img.shields.io/badge/AppVersion-4.2.3-informational?style=flat-square) @@ -40,61 +40,43 @@ Check the official documentation at **[docs.airlock.com](https://docs.airlock.co The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. ## Prerequisites -* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) -* [Airlock Microgateway License](#obtain-airlock-microgateway-license) -* [cert-manager](https://cert-manager.io/) * [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) -In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. -For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. -### Obtain Airlock Microgateway License -1. Either request a community or premium license - * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) - * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) -2. Check your inbox and save the license file microgateway-license.txt locally. - -> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. -### Deploy cert-manager -```bash -# Install cert-manager -kubectl apply -k https://github.com/airlock/microgateway/examples/utilities/cert-manager/?ref=4.2.3 - -# Wait for the cert-manager to be up and running -kubectl -n cert-manager wait --for=condition=ready --timeout=600s pod -l app.kubernetes.io/instance=cert-manager -``` - -## Deploy Airlock Microgateway Operator - -> This guide assumes a microgateway-license.txt file is present in the working directory. - -1. Install CRDs and Operator. +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). ```bash - # Create namespace - kubectl create namespace airlock-microgateway-system - - # Install License - kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt - - # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) - helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.2.3' --wait + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' -f https://raw.githubusercontent.com/airlock/microgateway/4.2.3/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' -f https://raw.githubusercontent.com/airlock/microgateway/4.2.3/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). 2. (Recommended) You can verify the correctness of the installation with `helm test`. ```bash - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.2.3' - helm test airlock-microgateway -n airlock-microgateway-system --logs - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.2.3' + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.2.3' ``` -### Upgrading CRDs - -The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. -CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: -```bash -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.2.3 --server-side --force-conflicts -``` - -**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. ## Support @@ -107,45 +89,33 @@ For the community edition, check our **[Airlock community forum](https://forum.a | Key | Type | Default | Description | |-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | | commonAnnotations | object | `{}` | Annotations to add to all resources. | | commonLabels | object | `{}` | Labels to add to all resources. | -| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | -| engine.image.digest | string | `"sha256:9b0debeef611172aa5ca79c6b8cd045e56a3c883763ec62c0fa211bb86d35304"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | -| engine.image.tag | string | `"4.2.3"` | Image tag to pull. | -| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | -| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | -| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | | fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:82b5924866840f783cce2e9b4095b7710a0e1cbf555498e8723ca811ca916290"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.2.3"` | Image tag to pull. | | imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | -| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | -| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | -| networkValidator.image.digest | string | `"sha256:a212cef6665b2464a41307162fa96e9623aa45c3fa32c39d320eae8b730d81e0"` | SHA256 image digest to pull (in the format "sha256:a212cef6665b2464a41307162fa96e9623aa45c3fa32c39d320eae8b730d81e0"). Overrides tag when specified. | -| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| networkValidator.image.repository | string | `"cgr.dev/chainguard/busybox"` | Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. | -| networkValidator.image.tag | string | `""` | Image tag to pull. | -| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | -| operator.config.logLevel | string | `"info"` | Operator application log level. | -| operator.image.digest | string | `"sha256:a429dfdb636e76bfbee7c59cfbe53d5f396c1f5603d5cb187f6283301ba4d7ba"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | -| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | -| operator.image.tag | string | `"4.2.3"` | Image tag to pull. | -| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | -| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | -| operator.podLabels | object | `{}` | Labels to add to all Pods. | -| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | -| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | -| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | -| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | -| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | -| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | -| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | -| operator.serviceLabels | object | `{}` | Labels to add to the Service. | -| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | -| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | -| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | -| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | | tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | ## License diff --git a/charts/airlock/microgateway/4.2.3/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/accesscontrols.microgateway.airlock.com.yaml deleted file mode 100644 index dea146ba5..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/accesscontrols.microgateway.airlock.com.yaml +++ /dev/null @@ -1,124 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: accesscontrols.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: AccessControl - listKind: AccessControlList - plural: accesscontrols - singular: accesscontrol - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControl specifies the options to perform access control with a Microgateway Engine container. - 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: Specifies how the Airlock Microgateway Engine performs access control. - properties: - policies: - description: Policies configures access control policies. - items: - properties: - authorization: - description: Authorization configures how requests are authorized. An empty object value {} disables authorization. - properties: - authentication: - description: Authentication specifies that clients need to be authenticated with the provided method. - properties: - oidc: - description: OIDC configures client authentication using OpenID Connect. - properties: - oidcRelyingPartyRef: - description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - oidcRelyingPartyRef - type: object - type: object - type: object - identityPropagation: - description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. - properties: - actions: - description: Actions specifies the propagation actions. - items: - properties: - identityPropagationRef: - description: IdentityPropagationRef selects an IdentityPropagation to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - identityPropagationRef - type: object - type: array - onFailure: - description: |- - OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: - _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. - enum: - - Pass - type: string - required: - - actions - - onFailure - type: object - required: - - authorization - type: object - maxItems: 1 - minItems: 1 - type: array - required: - - policies - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/contentsecurities.microgateway.airlock.com.yaml deleted file mode 100644 index e5f25bf30..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/contentsecurities.microgateway.airlock.com.yaml +++ /dev/null @@ -1,127 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: contentsecurities.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: ContentSecurity - listKind: ContentSecurityList - plural: contentsecurities - singular: contentsecurity - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: ContentSecurity specifies the options to secure an upstream web application with a Microgateway Engine container. - 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: Specifies the options to secure an upstream web application with a Microgateway Engine container. - properties: - apiProtection: - description: |- - APIProtection defines the relevant configurations to protect APIs. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - openAPIRef: - description: |- - OpenAPIRef selects the relevant OpenAPI configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - filter: - description: |- - Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests - to protect against various attack patterns. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - denyRulesRef: - description: |- - DenyRulesRef selects the relevant DenyRules configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - headerRewritesRef: - description: |- - HeaderRewritesRef selects the relevant HeaderRewrites. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - limitsRef: - description: |- - LimitsRef selects the relevant Limits configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - parserRef: - description: |- - ParserRef selects the relevant Parser configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/denyrules.microgateway.airlock.com.yaml deleted file mode 100644 index 00e680b58..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/denyrules.microgateway.airlock.com.yaml +++ /dev/null @@ -1,1508 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: denyrules.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: DenyRules - listKind: DenyRulesList - plural: denyrules - singular: denyrules - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - DenyRules configures request filtering using Airlock built-in and custom deny rules. - Deny rules establish a negative security model. They define prohibited patterns which, when a match is found in a request, lead to it being blocked from reaching the upstream web application. - To handle possible false positives, lower the security level or define fine-granular deny rule exceptions - If undefined, default settings are applied, designed to work with most upstream web application services. - 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: Specification of the desired deny rules behavior. - properties: - request: - description: Request configures deny rules for downstream requests. - properties: - builtIn: - description: BuiltIn configures the built-in deny rules. - properties: - exceptions: - description: Exceptions allows to define exceptions for specific requests and deny rules. - items: - description: |- - DenyRulesException defines an exception for deny rules. Exceptions may be defined by any or a combination of the following elements: blockedData (the request data causing a block) or requestConditions (properties of a request without taking into consideration the reason why a request has been blocked). - At least one of blockedData and requestConditions must be set. - properties: - blockedData: - description: BlockedData defines an exception based on the request data causing the block. - properties: - header: - description: |- - Header defines an exception based on a blocked header. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON defines an exception based on a blocked JSON property. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - jsonPath: - description: |- - JSONPath defines the JSONPath pattern to match the path within the JSON. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - key: - description: |- - Key defines the key of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter defines an exception based on a blocked parameter. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source defines the source of the parameter. - enum: - - Query - - Post - - Any - type: string - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path defines an exception based on the blocked path. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment defines an exception based on a blocked path segment. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - segments: - description: Segments defines the position of a segment within the path. - properties: - index: - description: Index specifies an exact path segment position by index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value defines the value of a path segment. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - type: object - requestConditions: - description: RequestConditions defines an exception based on a property of a request without taking into consideration the reason why a request has been blocked. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKeys: - description: RuleKeys restricts the exception to a set of deny rules. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - type: object - type: array - overrides: - description: Overrides allows to override the builtIn settings for specific deny rules. - items: - description: DenyRulesOverride allows to override the builtIn settings for specific deny rules. - properties: - conditions: - description: Conditions select which built-in deny rules' settings will be adjusted. - properties: - ruleKeys: - description: RuleKeys is a list of built-in deny rule names. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - types: - description: Types defines the type of attributes the override should be applied on. If Types are defined without any RuleKeys the override is applied to all deny rules. - items: - description: |- - A deny rule override type name can be any of the following values: - Header | - Parameter | - Path | - JSON - enum: - - Header - - Parameter - - Path - - PathSegment - - JSON - type: string - minItems: 0 - type: array - type: object - settings: - description: Settings override the corresponding properties for the selected rules. - properties: - level: - description: Level specifies the filter strength. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: array - settings: - description: Settings contains the keys which will be adjusted. - properties: - level: - default: Standard - description: Level represents a set of deny rules with different filter strengths. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - type: object - type: object - custom: - description: Custom allows configuring additional deny rules. - properties: - rules: - description: Rules defines list of additional deny rules. - items: - properties: - blockData: - description: BlockData specifies the request data which should cause a block. - properties: - header: - description: |- - Header specifies to block requests containing a matching header. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON specifies to block requests containing a matching JSON property in the body. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - key: - description: Key defines the key of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter specifies to block requests containing a matching parameter. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path specifies to block requests with a matching path. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - matcher: - description: Matcher specifies which path to block. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment specifies to block requests containing a matching path segment. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - segments: - description: |- - Segments restricts which path segments are filtered by this rule. - If not specified, all segments of a path are filtered. - properties: - index: - description: Index restricts the rule to the path segment at this index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value specifies which path segment values to block. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - value - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKey: - description: RuleKey defines a technical key for the deny rule. Must be unique. - minLength: 1 - pattern: ^[A-Z][A-Z0-9_]*$ - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - required: - - blockData - - ruleKey - type: object - type: array - x-kubernetes-list-map-keys: - - ruleKey - x-kubernetes-list-type: map - type: object - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/envoyclusters.microgateway.airlock.com.yaml deleted file mode 100644 index 90983db5d..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/envoyclusters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: envoyclusters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyCluster - listKind: EnvoyClusterList - plural: envoyclusters - singular: envoycluster - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyCluster is an additional Envoy Cluster resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy cluster. - properties: - value: - description: Value defines the Envoy Cluster which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/envoyconfigurations.microgateway.airlock.com.yaml deleted file mode 100644 index 6b73e0bfc..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/envoyconfigurations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,182 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: envoyconfigurations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyConfiguration - listKind: EnvoyConfigurationList - plural: envoyconfigurations - singular: envoyconfiguration - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - EnvoyConfiguration is the Schema for the envoyconfigurations API - {{% notice warning %}} EnvoyConfiguration resources may contain sensitive information and thus RBAC permissions should be granted with care. {{% /notice %}} - 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: EnvoyConfigurationSpec defines the desired state of EnvoyConfiguration - properties: - envoyResources: - description: EnvoyResources defines the desired state for each resource type. - properties: - clusters: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - endpoints: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - extensions: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - listeners: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - routes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - runtimes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - scopedRoutes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - secrets: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - nodeID: - description: NodeID defines the ID of the envoy node - type: string - required: - - nodeID - type: object - status: - description: EnvoyConfigurationStatus defines the observed state of EnvoyConfiguration - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of EnvoyConfiguration condition. - type: string - required: - - status - - type - type: object - type: array - status: - type: string - xds: - properties: - resourceTypes: - additionalProperties: - description: XdsResourceTypeSyncStatus defines the sync status of xDS for a specific resource type - properties: - errorMessage: - description: ErrorMessage defines an optional message why the currently served resources of this resource type are rejected by the client. - type: string - resources: - additionalProperties: - description: XdsResourceStatus defines the status of xDS for a specific resource - properties: - version: - description: Version defines the version which is currently served for this resource. - type: string - required: - - version - type: object - description: Resources defines the resources which are currently served for this resource type. - type: object - status: - description: Status defines the current sync status of this resource type. - type: string - version: - description: Version defines the version which is currently served for this resource type. - type: string - required: - - resources - - status - - version - type: object - description: ResourceTypes defines the sync statuses for each resource type. - type: object - version: - description: Version defines the version of the underlying xDS snapshot. - type: integer - required: - - version - type: object - required: - - status - - xds - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/headerrewrites.microgateway.airlock.com.yaml deleted file mode 100644 index 9df1a1db2..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/headerrewrites.microgateway.airlock.com.yaml +++ /dev/null @@ -1,759 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: headerrewrites.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: HeaderRewrites - listKind: HeaderRewritesList - plural: headerrewrites - singular: headerrewrites - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. - properties: - request: - description: Request defines manipulations on upstream request headers. - properties: - add: - description: Add defines which request headers will be added before forwarding to the upstream. - properties: - custom: - description: |- - Custom allows configuring additional upstream request headers. - Add selected headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which request headers will be forwarded to the upstream. - This can either be allHeaders or matchingHeaders. - Default: matchingHeaders: {...} - properties: - allHeaders: - description: AllHeaders specifies that all request headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which request headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - standardHeaders: - default: true - description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which request headers will be removed before forwarding to the upstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - alternativeForwardedHeaders: - default: true - description: |- - AlternativeForwardedHeaders removes downstream request headers which could potentially - be abused to alter the upstream's view of the remote connection. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - response: - description: Response defines manipulations on upstream response headers. - properties: - add: - description: Add defines which response headers will be added before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - csp: - default: true - description: |- - CSP sets a content security policy which allows only same-origin requests except for images - if the 'Content-Security-Policy' header is not set by the upstream. - type: boolean - featurePolicy: - default: false - description: |- - FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features - if the 'Feature-Policy' header is not set by the upstream. - **Deprecated:** Use permissionsPolicy instead. - type: boolean - hsts: - default: true - description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. - type: boolean - hstsPreload: - default: false - description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. - type: boolean - permissionsPolicy: - default: true - description: |- - PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features - if the 'Permissions-Policy' header is not set by the upstream. - type: boolean - referrerPolicy: - default: true - description: |- - ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests - if the 'Referrer-Policy' header is not set by the upstream. - type: boolean - xContentTypeOptions: - default: true - description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. - type: boolean - xFrameOptions: - default: true - description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which response headers will be forwarded to the downstream. - This can either be allHeaders or matchingHeaders. - Default: allHeaders: {} - properties: - allHeaders: - description: AllHeaders specifies that all response headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which response headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response header. - properties: - standardHeaders: - default: false - description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which response headers will be removed before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - auth: - description: Auth defines the categories of headers concerning authentication. - properties: - basic: - default: false - description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. - type: boolean - negotiate: - default: true - description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. - type: boolean - ntlm: - default: true - description: |- - NTLM removes upstream response headers that advise clients to authenticate with NTLM. - By default, these headers are removed, because NTLM pass-through is not supported. - type: boolean - type: object - informationLeakage: - description: InformationLeakage defines the categories of headers concerning information leakage. - properties: - application: - default: true - description: Application removes upstream response headers that leak information about the deployed software. - type: boolean - server: - default: true - description: Server removes upstream response headers that leak information about the server. - type: boolean - type: object - permissiveCors: - default: true - description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured remove operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - settings: - description: Settings configures the HeaderRewrites filter. - properties: - operationalMode: - default: Production - description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. - enum: - - Production - - Integration - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.2.3/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/identitypropagations.microgateway.airlock.com.yaml deleted file mode 100644 index 9b8c69599..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/identitypropagations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,108 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: identitypropagations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: IdentityPropagation - listKind: IdentityPropagationList - plural: identitypropagations - singular: identitypropagation - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: IdentityPropagation specifies the desired identity propagation. - 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: Specification of the desired identity propagation. - properties: - header: - description: Header configures identity propagation via a request header. - properties: - name: - description: Name of the header to set. - minLength: 1 - type: string - value: - description: Value to propagate to the application. - properties: - source: - description: Source from which to extract the value. - properties: - metadata: - description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. - properties: - key: - description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. - minLength: 1 - type: string - namespace: - description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. - minLength: 1 - type: string - required: - - key - - namespace - type: object - oidc: - description: OIDC specifies to extract a value from the result of an OpenID Connect flow. - properties: - idToken: - description: IDToken specifies to extract the value from the OpenID Connect ID Token. - properties: - claim: - description: Claim selects the JWT claim from which to extract the value. - minLength: 1 - type: string - required: - - claim - type: object - required: - - idToken - type: object - type: object - required: - - source - type: object - required: - - name - - value - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/limits.microgateway.airlock.com.yaml deleted file mode 100644 index 23adafe3b..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/limits.microgateway.airlock.com.yaml +++ /dev/null @@ -1,453 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: limits.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Limits - listKind: LimitsList - plural: limits - singular: limits - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Limits contains the configuration for limits. - 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: Specification of the desired limits behavior. - properties: - request: - description: Request defines the limits for requests. - properties: - limited: - description: Limited enables limits on request scope. - properties: - exceptions: - description: Exceptions defines limit exceptions. - items: - description: LimitsException defines an exception for limits. - properties: - length: - description: Length defines an exception for length limits based on the data element exceeding the limit. - properties: - json: - description: JSON defines a key and value length limit exception for a JSON property. - properties: - jsonPath: - description: |- - JSONPath restricts the exception to JSON properties with a matching JSONPath. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - required: - - jsonPath - type: object - parameter: - description: Parameter defines a name and value length limit exception for a parameter. - properties: - name: - description: Name restricts the exception to parameters with a matching name. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source restricts the exception to parameters of this kind. - enum: - - Query - - Post - - Any - type: string - required: - - name - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this exception to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - type: object - type: array - general: - description: General defines general request limits. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the request body. It specifies the number of bytes (0 = unlimited). This limit is effective only for requests that are parsed (e.g. JSON data). File uploads are not affected by this limit. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - pathLength: - anyOf: - - type: integer - - type: string - default: 1Ki - description: PathLength defines the maximum path length for requests. - 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 - json: - description: JSON defines the limits for JSON requests. - properties: - elementCount: - default: 10000 - description: ElementCount defines the maximum number of keys and array items in the whole JSON document (recursive). - format: int64 - type: integer - keyCount: - default: 250 - description: KeyCount defines the maximum number of keys of a single JSON object (non-recursive). - format: int64 - type: integer - keyLength: - anyOf: - - type: integer - - type: string - default: "128" - description: KeyLength defines the maximum length for JSON keys. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - nestingDepth: - default: 100 - description: NestingDepth defines the maximum depth of nesting for JSON objects and JSON arrays. - format: int64 - type: integer - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for JSON values. - 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 - parameter: - description: Parameter defines the limits for request parameters. - properties: - count: - default: 128 - description: Count defines the maximum number of request parameters. - format: int64 - type: integer - nameLength: - anyOf: - - type: integer - - type: string - default: "128" - description: NameLength defines the maximum length for parameter names. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for parameter values. - 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 - unlimited: - description: Unlimited disables all limits on request scope. - type: object - type: object - settings: - description: Settings configures the limits filter. - properties: - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a limit hits. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.2.3/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/oidcproviders.microgateway.airlock.com.yaml deleted file mode 100644 index 69116b40a..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/oidcproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,301 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: oidcproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCProvider - listKind: OIDCProviderList - plural: oidcproviders - singular: oidcprovider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCProvider specifies an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - ID token and access token are stored in cookies and are thus sent to the accessing client. - {{% /notice %}} - 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: Specification of an OpenID Provider. - properties: - static: - description: Static configures an OpenID Provider by explicitly specifying all endpoints. - properties: - endpoints: - description: Endpoints specifies the OpenID Provider endpoints. - properties: - authorization: - description: Authorization specifies the endpoint to which the authorization request is sent. - properties: - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - token: - description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. - properties: - tls: - description: TLS defines TLS settings. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: |- - Custom explicitly specifies how the server certificate should be verified. - Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines constraints the presented certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - disabled: - description: |- - Disabled specifies to trust any certificate without verification. - THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. - type: object - type: object - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - required: - - authorization - - token - type: object - required: - - endpoints - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml deleted file mode 100644 index 708e48aaf..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml +++ /dev/null @@ -1,219 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: oidcrelyingparties.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCRelyingParty - listKind: OIDCRelyingPartyList - plural: oidcrelyingparties - singular: oidcrelyingparty - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - ID token and access token are stored in cookies and are thus sent to the accessing client. - {{% /notice %}} - 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: Specification of the OIDC Relying Party configuration. - properties: - clientID: - description: ClientID specifies the OIDCRelyingParty "client_id". - minLength: 1 - type: string - credentials: - description: Credentials used for client authentication on the back-channel with the authorization server. - properties: - clientSecret: - description: ClientSecret authenticates with the client password issued by the OpenID Provider (OP). - properties: - method: - default: BasicAuth - description: Method specifies in which format the client secret is sent with the authorization request. - enum: - - BasicAuth - - FormURLEncoded - type: string - secretRef: - description: SecretRef specifies the kubernetes secret containing the client password with key "client.secret". - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - clientSecret - type: object - oidcProviderRef: - description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - pathMapping: - description: PathMapping configures the action matching. - properties: - logoutPath: - description: LogoutPath specifies which request paths should initiate a logout. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - redirectPath: - description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - logoutPath - - redirectPath - type: object - redirectURI: - description: |- - RedirectURI configures the "redirect_uri" parameter included in the authorization request. - May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. - minLength: 1 - type: string - required: - - clientID - - credentials - - oidcProviderRef - - pathMapping - - redirectURI - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/openapis.microgateway.airlock.com.yaml deleted file mode 100644 index fafda725e..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/openapis.microgateway.airlock.com.yaml +++ /dev/null @@ -1,167 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: openapis.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OpenAPI - listKind: OpenAPIList - plural: openapis - singular: openapi - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: OpenAPI contains the configuration for the OpenAPI specification. - 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: Specification of the desired OpenAPI specification. - properties: - response: - description: Response defines the validation behaviour for responses. - properties: - secured: - description: Secured enables response checking. - properties: - validation: - default: Lax - description: Validation defines the validation mode for responses. - enum: - - Lax - - Strict - type: string - type: object - unsecured: - description: Unsecured disables response checking. - type: object - type: object - settings: - description: Settings defines the settings to configure OpenAPI specification enforcement. - properties: - logging: - description: Logging specifies the access log behavior. - properties: - maxFailedSubvalidations: - default: 10 - description: MaxFailedSubvalidations defines the maximum number of failed subvalidations being logged. - format: int64 - type: integer - type: object - schema: - description: Schema configures the OpenAPI specification. - properties: - source: - description: Source specifies the OpenAPI specification to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'openapi.json'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - validation: - description: Validation specifies the patterns for the validation behavior. - properties: - authentication: - description: Authentication defines the settings for the authentication scheme. - properties: - oAuth2: - description: OAuth2 specifies the OAuth2 parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - oidc: - description: Oidc specifies the OIDC parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - type: object - type: object - required: - - schema - type: object - required: - - settings - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.2.3/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/parsers.microgateway.airlock.com.yaml deleted file mode 100644 index b450d488b..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/parsers.microgateway.airlock.com.yaml +++ /dev/null @@ -1,358 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: parsers.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Parser - listKind: ParserList - plural: parsers - singular: parser - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Parser contains the configuration for content parsers (default and custom). - 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: Specification of the desired parser behavior. - properties: - request: - description: Request defines the parsing for downstream requests. - properties: - custom: - description: Custom allows configuring additional rules for parser selection. - properties: - rules: - description: |- - Rules defines a custom set prepended before built-in rules of enabled request parsers. - Disable all built-in parsers to overrule them completely. - items: - properties: - action: - description: |- - Action specifies what should happen when a request condition matches. - Only one of parse or skip can be set. - properties: - parse: - description: Parse activates the configured parser. - properties: - form: - description: Form activates the Form parser. - type: object - json: - description: JSON activates the JSON parser. - type: object - multipart: - description: Multipart activates the multipart parser. - type: object - type: object - skip: - description: Skip disables any content parsing - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - required: - - action - - requestConditions - type: object - type: array - type: object - defaultContentType: - default: application/x-www-form-urlencoded - description: DefaultContentType specifies the content-type header which should be injected into the request before parser selection if it is not already present and the request has a body. - minLength: 1 - type: string - parsers: - description: Parsers defines the configuration for the available content parsers. - properties: - form: - description: Form defines the configuration for the form parser. - properties: - enable: - default: true - description: Enable defines whether form payloads are inspected. - type: boolean - mediaTypePattern: - default: .*urlencoded.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as form arguments. - minLength: 1 - type: string - type: object - json: - description: JSON defines the configuration for the JSON parser. - properties: - enable: - default: true - description: Enable defines whether json payloads are inspected. - type: boolean - mediaTypePattern: - default: .*json.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as JSON. - minLength: 1 - type: string - type: object - multipart: - description: Multipart defines the configuration for the multipart parser. - properties: - enable: - default: true - description: Enable defines whether multipart payloads are inspected. - type: boolean - mediaTypePattern: - default: .*multipart.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as a multipart payload. - minLength: 1 - type: string - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.2.3/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/sidecargateways.microgateway.airlock.com.yaml deleted file mode 100644 index 04b5f45d3..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/sidecargateways.microgateway.airlock.com.yaml +++ /dev/null @@ -1,731 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: sidecargateways.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SidecarGateway - listKind: SidecarGatewayList - plural: sidecargateways - singular: sidecargateway - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: SidecarGateway contains the configuration how to configure the Airlock Microgateway Engine when used as Sidecar Container within the Pod of an application. - 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: Specification of the desired sidecar gateway behavior. - properties: - applications: - description: Applications defines applications which run on different ports. - items: - properties: - containerPort: - default: 8080 - description: |- - ContainerPort refers to the container port. - This must be a valid port number, 0 < x < 65536. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - downstream: - description: Downstream defines the downstream configuration for this application - properties: - protocol: - description: |- - Protocol defines the exposed HTTP protocol version. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies that the protocol should be inferred. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies that the client is assumed to speak HTTP/1.1. - type: object - http2: - description: HTTP2 specifies that the client is assumed to speak HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - remoteIP: - description: |- - RemoteIP defines how the remote IP of a client is propagated. - Default: xff: {...} - properties: - connectionIP: - description: ConnectionIP configures to use the source IP address of the direct downstream connection. - type: object - customHeader: - description: CustomHeader specifies to use a custom header for remote IP extraction. - properties: - headerName: - description: HeaderName specifies the name of the custom header containing the remote IP. - minLength: 1 - type: string - required: - default: true - description: Required specifies if the custom header is required. If true and not available the request will be rejected with 403. - type: boolean - required: - - headerName - type: object - xff: - description: XFF configures to use the standard 'X-Forwarded-For' header for IP extraction. - properties: - numTrustedHops: - default: 1 - description: NumTrustedHops specifies to extract the client's originating IP from the nth rightmost entry in the X-Forwarded-For header. With the default value of 1, the IP is extracted from the rightmost entry. - format: int32 - minimum: 1 - type: integer - type: object - type: object - requestNormalizations: - description: RequestNormalizations defines a set of normalization actions which are applied to the request before route matching. - properties: - mergeSlashes: - default: true - description: MergeSlashes ensures that adjacent slashes in the path are merged into one. - type: boolean - normalizePath: - default: true - description: NormalizePath ensures normalization according to RFC 3986 without case normalization. - type: boolean - type: object - restrictions: - description: Restrictions defines restrictions for downstream. - properties: - http: - description: HTTP defines limits for the HTTP protocol. - properties: - headersLength: - anyOf: - - type: integer - - type: string - default: 60Ki - description: HeadersLength defines maximum size of all request headers combined. Requests that exceed this limit will receive a 431 response. - 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 - timeouts: - description: Timeouts defines timeouts for downstream - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - default: 5m - description: |- - Idle defines the settings for the idle timeout when no data is sent or received. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - maxDuration: - default: 5m - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - requestHeaders: - default: 10s - description: |- - RequestHeaders defines the duration before all request headers must be received. - A value of 0 will completely disable the timeout. - Default: 10s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - clientCertificate: - description: |- - ClientCertificate defines the TLS settings for verification of client certificates. - At most one of ignored, optional and required can be set. - Default: ignored: {} - properties: - ignored: - description: Ignored disables verification of the client certificate. - type: object - optional: - description: |- - Optional enables verification of the client certificate if one is presented. - In this mode only trustedCA and crl settings can be configured since certificatePinning and allowedSANs require a client certificate. - properties: - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - required: - - trustedCA - type: object - required: - description: |- - Required contains settings for client certificate verification. A client must present a valid certificate. - At least one of trustedCA and certificatePinning must be set. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines the constraints a client certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - type: object - enable: - default: false - description: Enable defines if the downstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - secretRef: - description: SecretRef defines the reference to the TLS server certificate (secret of type kubernetes.io/tls). - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - xfcc: - description: |- - XFCC defines the handling of X-Forwarded-Client-Cert header. Meaning of the possible values: - _Sanitize_: Do not send the XFCC header to the next hop. This is the default value. - _ForwardOnly_: When the client connection is mTLS (Mutual TLS), forward the XFCC header in the request. - _AppendAndForward_: When the client connection is mTLS, append the client certificate information to the request’s XFCC header and forward it. - _SanitizeAndSet_: When the client connection is mTLS, reset the XFCC header with the client certificate information and send it to the next hop. - _AlwaysForwardOnly_: Always forward the XFCC header in the request, regardless of whether the client connection is mTLS. - Note: When forwarding the XFCC header in the request you might have to adjust the header length restrictions (See sidecargateway.spec.applications.downstream.restrictions.http) - enum: - - Sanitize - - ForwardOnly - - AppendAndForward - - SanitizeAndSet - - AlwaysForwardOnly - type: string - type: object - type: object - envoyHTTPFilterRefs: - description: EnvoyHTTPFilterRefs selects the relevant EnvoyHTTPFilters. - properties: - prepend: - description: Prepend selects the relevant EnvoyHTTPFilters which are added before those configured by the Airlock Microgateway. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - type: object - routes: - description: Routes defines the security configurations for different paths. The first matching route (from top to bottom) applies. - items: - description: |- - SidecarGatewayApplicationRoute defines the security configurations for different paths. - At most one of secured and unsecured can be set. - Default: secured: {...} - properties: - pathPrefix: - default: / - description: PathPrefix defines the path prefix used during route selection. - minLength: 1 - type: string - secured: - description: Secured enables WAF processing for this route. - properties: - accessControlRef: - description: |- - AccessControlRef selects the relevant AccessControl configuration resource. - If undefined, Airlock Microgateway does not perform any access control. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - contentSecurityRef: - description: |- - ContentSecurityRef selects the relevant ContentSecurity configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - unsecured: - description: |- - Unsecured disables all WAF functionality and therefore protection for this route. - WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. - type: object - type: object - type: array - x-kubernetes-list-map-keys: - - pathPrefix - x-kubernetes-list-type: map - telemetryRef: - description: |- - TelemetryRef selects the relevant Telemetry configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - upstream: - description: Upstream defines the upstream configuration for this application - properties: - protocol: - description: |- - Protocol defines HTTP protocol version used to communicate with the upstream. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies to use the protocol negotiated via TLS ALPN (if supported) or HTTP/1.1 as fallback. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies to use HTTP/1.1. - type: object - http2: - description: HTTP2 specifies to use HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - timeouts: - description: Timeouts defines the timeout settings. - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - description: |- - Timeout defines the settings for http timeouts. If this setting is not specified, the value of applications[].downstream.timeouts.http.idle is inherited. - A value of 0 will completely disable the timeout. - type: string - maxDuration: - default: 15s - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - Default: 15s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - enable: - default: false - description: Enable defines if the upstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - type: object - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - containerPort - x-kubernetes-list-type: map - envoyClusterRefs: - description: EnvoyClusterRefs selects the relevant EnvoyClusters. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - podSelector: - description: PodSelector defines to which Pods the configuration will be applied to. - properties: - matchLabels: - additionalProperties: - type: string - description: MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels. - type: object - type: object - required: - - applications - type: object - status: - description: Most recently observed status of the SidecarGateway which is populated by the system. This data is read-only and may not be up to date. - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of SidecarGateway condition. - type: string - required: - - status - - type - type: object - type: array - pods: - items: - properties: - envoyConfig: - description: EnvoyConfig indicates the name of the EnvoyConfig CR which references the SidecarGateway. - type: string - name: - description: Name indicates the name of the Pod which references the SidecarGateway. - type: string - required: - - name - type: object - type: array - status: - type: string - required: - - status - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.2.3/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.2.3/crds/telemetries.microgateway.airlock.com.yaml deleted file mode 100644 index b55fcba21..000000000 --- a/charts/airlock/microgateway/4.2.3/crds/telemetries.microgateway.airlock.com.yaml +++ /dev/null @@ -1,81 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.14.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 - name: telemetries.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Telemetry - listKind: TelemetryList - plural: telemetries - singular: telemetry - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Telemetry contains the configuration for telemetry (logging, metrics & tracing). - 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: Specification of the desired telemetry behavior. - properties: - correlation: - description: Correlation defines the correlation aspects of Telemetry. - properties: - request: - description: Request defines the request related correlation settings of Telemetry. - properties: - allowDownstreamRequestID: - default: true - description: AllowDownstreamRequestID defines whether trace sampling will consider a provided x-request-id. - type: boolean - alterRequestID: - default: true - description: AlterRequestID defines whether to alter the UUID to reflect the trace sampling decision. If disabled no modification to the UUID will be performed, this may break tracing in the upstream. - type: boolean - type: object - type: object - logging: - description: Logging defines the logging aspects of Telemetry. - properties: - accessLog: - description: AccessLog defines the access log settings of Telemetry. - properties: - format: - description: Format defines the Access Log format of the sidecar. - properties: - json: - description: JSON defines the Access Log format as JSON. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.2.3/gke-values.yaml b/charts/airlock/microgateway/4.2.3/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway/4.2.3/openshift-values.yaml b/charts/airlock/microgateway/4.2.3/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway/4.2.3/questions.yml b/charts/airlock/microgateway/4.2.3/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway/4.2.3/templates/NOTES.txt b/charts/airlock/microgateway/4.2.3/templates/NOTES.txt index 8e7d84a8f..e8aa45888 100644 --- a/charts/airlock/microgateway/4.2.3/templates/NOTES.txt +++ b/charts/airlock/microgateway/4.2.3/templates/NOTES.txt @@ -1,22 +1,3 @@ -Thank you for installing Airlock Microgateway. -If you have not already done so, make sure that Airlock Microgateway CNI is also installed on the cluster. +Thank you for installing Airlock Microgateway CNI. -For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}. -Detailed CRD API reference documentation is also available at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}/api/crds. -{{ if .Values.crds.skipVersionCheck }} -- CRD version check skipped -{{- else }} -{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} -{{- if $outdatedCRDs -}} - {{- fail (printf ` - -Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. -Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: - -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts - -If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` - .Chart.AppVersion) - -}} -{{- end -}} -{{- end -}} +For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" .}}. diff --git a/charts/airlock/microgateway/4.2.3/templates/_helpers.tpl b/charts/airlock/microgateway/4.2.3/templates/_helpers.tpl index 1c3bb34a2..996491a87 100644 --- a/charts/airlock/microgateway/4.2.3/templates/_helpers.tpl +++ b/charts/airlock/microgateway/4.2.3/templates/_helpers.tpl @@ -1,16 +1,14 @@ {{/* Expand the name of the chart. -We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest explicit suffix is 14 characters. */}} -{{- define "airlock-microgateway.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Convert an image configuration object into an image ref string. */}} -{{- define "airlock-microgateway.image" -}} +{{- define "airlock-microgateway-cni.image" -}} {{- if .digest -}} {{- printf "%s@%s" .repository .digest -}} {{- else if .tag -}} @@ -22,19 +20,19 @@ Convert an image configuration object into an image ref string. {{/* Create a default fully qualified app name. -We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest implicit suffix is 27 characters. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. If release name contains chart name it will be used as a full name. */}} -{{- define "airlock-microgateway.fullname" -}} +{{- define "airlock-microgateway-cni.fullname" -}} {{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} {{- else }} {{- $name := default .Chart.Name .Values.nameOverride }} {{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} {{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} {{- end }} {{- end }} {{- end }} @@ -42,88 +40,59 @@ If release name contains chart name it will be used as a full name. {{/* Create chart name and version as used by the chart label. */}} -{{- define "airlock-microgateway.chart" -}} +{{- define "airlock-microgateway-cni.chart" -}} {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Common labels */}} -{{- define "airlock-microgateway.sharedLabels" -}} -helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} {{- if .Chart.AppVersion }} app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} {{- end }} app.kubernetes.io/managed-by: {{ .Release.Service }} -app.kubernetes.io/part-of: {{ .Chart.Name }} {{- with .Values.commonLabels }} {{ toYaml .}} {{- end }} {{- end }} {{/* -Common Selector labels +Common labels without component */}} -{{- define "airlock-microgateway.sharedSelectorLabels" -}} -app.kubernetes.io/instance: {{ .Release.Name }} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} {{- end }} {{/* -Restricted Container Security Context +Selector labels */}} -{{- define "airlock-microgateway.restrictedSecurityContext" -}} -allowPrivilegeEscalation: false -privileged: false -runAsNonRoot: true -capabilities: - drop: ["ALL"] -readOnlyRootFilesystem: true -seccompProfile: - type: RuntimeDefault +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} {{- end }} -{{/* Precondition: May only be used if AppVersion is isSemver */}} -{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} -{{- $version := (semver .Chart.AppVersion) -}} -{{- if $version.Prerelease -}} ->= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} -{{- else -}} ->= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 -{{- end -}} -{{- end -}} +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} -{{- define "airlock-microgateway.outdatedCRDs" -}} -{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} - {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} - {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} - {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} - {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} - {{- $isOutdated := false -}} - {{- if $crd -}} - {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} - {{- $isOutdated = true -}} - {{- if hasKey $crd.metadata "labels" -}} - {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} - {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} - {{- if (semverCompare $supportedVersion $crdVersion) }} - {{- $isOutdated = false -}} - {{- end }} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if $isOutdated }} -{{ base $path }} - {{- end }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{- define "airlock-microgateway.isSemver" -}} +{{- define "airlock-microgateway-cni.isSemver" -}} {{- regexMatch `^(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-]+)*))?$` . -}} {{- end -}} -{{- define "airlock-microgateway.docsVersion" -}} -{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} {{- $version := (semver .Chart.AppVersion) -}} {{- $version.Major }}.{{ $version.Minor -}} {{- else -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/clusterrole.yaml b/charts/airlock/microgateway/4.2.3/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/clusterrolebinding.yaml b/charts/airlock/microgateway/4.2.3/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/configmap.yaml b/charts/airlock/microgateway/4.2.3/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway/4.2.3/templates/daemonset.yaml b/charts/airlock/microgateway/4.2.3/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway/4.2.3/templates/network-attachment-definition.yaml b/charts/airlock/microgateway/4.2.3/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.2.3/templates/operator/_webhooks.gen.tpl deleted file mode 100644 index 3c12e34e7..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/_webhooks.gen.tpl +++ /dev/null @@ -1,299 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator mutating webhooks -*/}} -{{- define "airlock-microgateway-operator.mutatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /mutate-v1-pod - failurePolicy: Fail - name: mutate-pod.microgateway.airlock.com - reinvocationPolicy: IfNeeded - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -{{- end }} - -{{/* -Operator validating webhooks -*/}} -{{- define "airlock-microgateway-operator.validatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-accesscontrol - failurePolicy: Fail - name: validate-accesscontrol.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - accesscontrols - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-denyrules - failurePolicy: Fail - name: validate-denyrules.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - denyrules - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoycluster - failurePolicy: Fail - name: validate-envoycluster.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyclusters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoyhttpfilter - failurePolicy: Fail - name: validate-envoyhttpfilter.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyhttpfilters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-headerrewrites - failurePolicy: Fail - name: validate-headerrewrites.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - headerrewrites - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-identitypropagation - failurePolicy: Fail - name: validate-identitypropagation.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - identitypropagations - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-limits - failurePolicy: Fail - name: validate-limits.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - limits - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcprovider - failurePolicy: Fail - name: validate-oidcprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcrelyingparty - failurePolicy: Fail - name: validate-oidcrelyingparty.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcrelyingparties - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-openapi - failurePolicy: Fail - name: validate-openapi.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - openapis - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-parser - failurePolicy: Fail - name: validate-parser.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - parsers - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-sidecargateway - failurePolicy: Fail - name: validate-sidecargateway.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - sidecargateways - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-v1-pod - failurePolicy: Fail - name: validate-pod.microgateway.airlock.com - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - - UPDATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -{{- end }} diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/configmap.yaml deleted file mode 100644 index 113d8a47b..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/configmap.yaml +++ /dev/null @@ -1,322 +0,0 @@ -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -data: - engine_bootstrap_config_template.yaml: | - # Base configuration, admin interface on port 19000 - admin: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - dynamic_resources: - cds_config: - initial_fetch_timeout: 10s - resource_api_version: V3 - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - lds_config: - resource_api_version: V3 - initial_fetch_timeout: 10s - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - static_resources: - listeners: - - name: probe - address: - socket_address: - address: 0.0.0.0 - port_value: 19001 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: http - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: probe - virtual_hosts: - - name: probe - domains: - - '*' - routes: - - name: ready - match: - path: /ready - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - - name: metrics - address: - socket_address: - address: 0.0.0.0 - port_value: 19002 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: http - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: metrics - virtual_hosts: - - name: metrics - domains: - - '*' - routes: - - name: metrics - match: - path: /metrics - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - prefix_rewrite: '/stats/prometheus' - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - clusters: - - name: xds_cluster - connect_timeout: 1s - type: STRICT_DNS - load_assignment: - cluster_name: xds_cluster - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: airlock-microgateway-operator-xds.$(OPERATOR_NAMESPACE).svc.cluster.local - port_value: 13377 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - transport_socket: - name: envoy.transport_sockets.tls - typed_config: - '@type': type.googleapis.com/envoy.extensions.transport_sockets.tls.v3.UpstreamTlsContext - common_tls_context: - tls_params: - tls_minimum_protocol_version: TLSv1_3 - tls_maximum_protocol_version: TLSv1_3 - validation_context_sds_secret_config: - name: validation_context_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/validation_context_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - tls_certificate_sds_secret_configs: - - name: tls_certificate_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/tls_certificate_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - - name: airlock_microgateway_engine_admin - connect_timeout: 1s - type: STATIC - load_assignment: - cluster_name: airlock_microgateway_engine_admin - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - stats_config: - stats_tags: - - tag_name: "category" - regex: "\\.(category\\.([^.]+))" - - tag_name: "rule_name" - regex: "\\.(rule\\.([^.]+))" - - tag_name: "limit_name" - regex: "\\.(limit\\.([^.]+))" - - tag_name: "threat_handling_mode" - regex: "\\.(threat_handling_mode\\.([^.]+))" - - tag_name: "envoy_cluster_name" - regex: "\\.(cluster\\.([^.]+))" - - tag_name: "version" - regex: "\\.(version\\.([^.]+))" - use_all_default_tags: true - bootstrap_extensions: - - name: airlock.bootstrap.engine_build_info - typed_config: - '@type': type.googleapis.com/airlock.extensions.bootstrap.stats.v1alpha.Stats - application_log_config: - log_format: - text_format: '{"@timestamp":"%Y-%m-%dT%T.%e%z","log":{"logger":"%n","level":"%l","origin":{"file":{"name":"%g","line":%#},"function":"%!"}},"event":{"module":"envoy","dataset":"envoy.application"},"process":{"pid":%P,"thread":{"id":%t}},"ecs":{"version":"8.5"},"message":"%j"}' - engine_container_template.yaml: | - name: "$(ENGINE_NAME)" - image: "$(ENGINE_IMAGE)" - imagePullPolicy: {{ .Values.engine.image.pullPolicy }} - args: - - "--config-path" - - "/etc/envoy/bootstrap_config.yaml" - - "--base-id" - - "$(BASE_ID)" - - "--file-flush-interval-msec" - - '1000' - - "--drain-time-s" - - '60' - - "--service-node" - - "$(POD_NAME).$(POD_NAMESPACE)" - - "--service-cluster" - - "$(APP_NAME).$(POD_NAMESPACE)" - - "--log-path" - - "/dev/stdout" - - "--log-level" - - "$(LOG_LEVEL)" - volumeMounts: - - name: airlock-microgateway-bootstrap-secret-volume - mountPath: /etc/envoy - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - - name: POD_IP - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: status.podIP - ports: - - containerPort: 13378 - protocol: TCP - - containerPort: 19001 - protocol: TCP - - containerPort: 19002 - protocol: TCP - livenessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 1 - readinessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.engine.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - network_validator_container_template.yaml: | - name: "$(NETWORK_VALIDATOR_NAME)" - image: "$(NETWORK_VALIDATOR_IMAGE)" - imagePullPolicy: {{ .Values.networkValidator.image.pullPolicy }} - command: ["/bin/sh", "-c"] - args: - - |- - echo 'pong' | nc -v -l 127.0.0.1 -p 13378 & - for i in 1 2 3; do - sleep 1s - if r=$(echo 'ping' | nc 127.0.0.1 19003) && [ $r == pong ]; then - echo -n 'Traffic redirection to Airlock Microgateway Engine is working.' > /dev/termination-log - exit 0 - fi - done - echo -en 'Traffic redirection to Airlock Microgateway Engine is not working.\nRestart the pod after ensuring that hostNetwork is disabled and a compatible Airlock Microgateway CNI version is installed on the node.\nCertain environments may also require additional configuration (see docs.airlock.com for more information).' > /dev/termination-log - exit 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - operator_config.yaml: | - apiVersion: config.airlock.com/v1alpha1 - kind: OperatorConfig - health: - healthProbeBindAddress: :8081 - metrics: - bindAddress: 0.0.0.0:8080 - webhook: - port: 9443 - deployment: - sidecar: - engineContainerTemplate: "/sidecar/engine_container_template.yaml" - networkValidatorContainerTemplate: "/sidecar/network_validator_container_template.yaml" - engine: - bootstrapConfigTemplate: "/engine_bootstrap_config_template.yaml" - log: - level: {{ .Values.operator.config.logLevel }} diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/deployment.yaml deleted file mode 100644 index a14cd9bd3..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/deployment.yaml +++ /dev/null @@ -1,138 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.operator.replicaCount }} - {{- with .Values.operator.updateStrategy }} - strategy: - {{- toYaml . | trim | nindent 4 }} - {{- end }} - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - checksum/config: {{ include (print $.Template.BasePath "/operator/configmap.yaml") . | sha256sum }} - kubectl.kubernetes.io/default-container: manager - {{- with mustMerge .Values.operator.podAnnotations .Values.commonAnnotations}} - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 8 }} - {{- with .Values.operator.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - containers: - - args: - - --config=operator_config.yaml - env: - - name: ENGINE_IMAGE - value: {{ include "airlock-microgateway.image" .Values.engine.image }} - - name: NETWORK_VALIDATOR_IMAGE - value: {{ include "airlock-microgateway.image" .Values.networkValidator.image }} - - name: OPERATOR_NAME - valueFrom: - fieldRef: - fieldPath: metadata.name - - name: OPERATOR_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - image: {{ include "airlock-microgateway.image" .Values.operator.image }} - imagePullPolicy: {{ .Values.operator.image.pullPolicy }} - livenessProbe: - httpGet: - path: /healthz - port: 8081 - initialDelaySeconds: 15 - periodSeconds: 20 - timeoutSeconds: 5 - name: manager - ports: - - containerPort: 9443 - name: webhook-server - protocol: TCP - - containerPort: 13377 - name: xds-server - protocol: TCP - - containerPort: 8080 - protocol: TCP - - containerPort: 8081 - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: 8081 - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - {{- with .Values.operator.resources }} - resources: - {{- toYaml . | nindent 10 }} - {{- end }} - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 10 }} - terminationMessagePath: /dev/termination-log - terminationMessagePolicy: FallbackToLogsOnError - volumeMounts: - - mountPath: /tmp/k8s-webhook-server/serving-certs - name: cert - readOnly: true - - mountPath: /opt/airlock/license/ - name: airlock-microgateway-license - readOnly: true - - mountPath: /operator_config.yaml - name: operator-config - subPath: operator_config.yaml - - mountPath: /sidecar/engine_container_template.yaml - name: operator-config - subPath: engine_container_template.yaml - - mountPath: /sidecar/network_validator_container_template.yaml - name: operator-config - subPath: network_validator_container_template.yaml - - mountPath: /engine_bootstrap_config_template.yaml - name: operator-config - subPath: engine_bootstrap_config_template.yaml - securityContext: - runAsNonRoot: true - serviceAccountName: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - terminationGracePeriodSeconds: 10 - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - name: cert - secret: - defaultMode: 420 - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert - - name: airlock-microgateway-license - secret: - defaultMode: 292 - optional: true - secretName: {{ .Values.license.secretName }} - - configMap: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - name: operator-config diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrole.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrole.yaml deleted file mode 100644 index d3ce5540c..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrole.yaml +++ /dev/null @@ -1,14 +0,0 @@ -{{- if .Values.operator.rbac.create }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" . -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrolebinding.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrolebinding.yaml deleted file mode 100644 index 059c161ea..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/manager-clusterrolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/mutating-webhook.yaml deleted file mode 100644 index 4583e1452..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/mutating-webhook.yaml +++ /dev/null @@ -1,14 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{ include "airlock-microgateway-operator.mutatingWebhooks" . -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/podmonitor.yaml deleted file mode 100644 index f84031633..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/podmonitor.yaml +++ /dev/null @@ -1,26 +0,0 @@ -{{- if .Values.engine.sidecar.podMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ include "airlock-microgateway.fullname" . }}-engine - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.engine.sidecar.podMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - any: true - selector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" - podMetricsEndpoints: - - targetPort: 19002 - path: /metrics - scheme: http -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.2.3/templates/operator/validating-webhook.yaml deleted file mode 100644 index 6332a0296..000000000 --- a/charts/airlock/microgateway/4.2.3/templates/operator/validating-webhook.yaml +++ /dev/null @@ -1,14 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{ include "airlock-microgateway-operator.validatingWebhooks" . -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.2.3/templates/scc-role.yaml b/charts/airlock/microgateway/4.2.3/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.2.3/templates/scc-rolebinding.yaml b/charts/airlock/microgateway/4.2.3/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/serviceaccount.yaml b/charts/airlock/microgateway/4.2.3/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway/4.2.3/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.2.3/templates/tests/rbac.yaml index a067a4304..744799333 100644 --- a/charts/airlock/microgateway/4.2.3/templates/tests/rbac.yaml +++ b/charts/airlock/microgateway/4.2.3/templates/tests/rbac.yaml @@ -2,106 +2,63 @@ apiVersion: v1 kind: ServiceAccount metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests --- apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests roleRef: apiGroup: rbac.authorization.k8s.io kind: Role - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" subjects: - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" --- apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests rules: - apiGroups: - - microgateway.airlock.com + - "apps" resources: - - sidecargateways + - daemonsets resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + - {{ include "airlock-microgateway-cni.fullname" . }} verbs: - - get - - list - - watch - - delete + - get + - watch + - list - apiGroups: - - microgateway.airlock.com + - "" resources: - - sidecargateways + - pods + - pods/log verbs: - - create + - get + - list +{{- if .Values.rbac.createSCCRole }} - apiGroups: - - "" - resources: - - events - verbs: - - list -- apiGroups: - - "apps" - resources: - - deployments + - security.openshift.io resourceNames: - - "{{ include "airlock-microgateway.operator.fullname" . }}" - verbs: - - get - - list - - watch -- apiGroups: - - "apps" + - privileged resources: - - statefulsets - - statefulsets/scale - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + - securitycontextconstraints verbs: - - get - - list - - watch - - patch -- apiGroups: - - "" - resources: - - pods - - pods/log - - pods/status - - pods/attach - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" - - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" - - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" - verbs: - - get - - list - - create - - watch - - delete -- apiGroups: - - "" - resources: - - pods - verbs: - - create + - use +{{- end -}} {{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.2.3/templates/tests/test-install.yaml index 35fd0f4e1..12d8c8de7 100644 --- a/charts/airlock/microgateway/4.2.3/templates/tests/test-install.yaml +++ b/charts/airlock/microgateway/4.2.3/templates/tests/test-install.yaml @@ -2,13 +2,11 @@ apiVersion: v1 kind: Pod metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" namespace: {{ .Release.Namespace }} labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} annotations: helm.sh/hook: test helm.sh/hook-delete-policy: before-hook-creation @@ -18,183 +16,88 @@ spec: - name: test image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true command: - sh - -c - | set -eu - clean_up() { - echo "" - echo "### Clean up test resources" - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - echo "" - echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=30s - sleep 3s - echo "" - } - fail() { + echo "Error: ${1}" echo "" - echo "### Error: ${1}" - echo "" - echo 'Microgateway Sidecargateway status:' - kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true - echo "" - echo "" - echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" - kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true - echo "" - echo "" - echo 'Logs of Nginx container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true - echo "" - echo "" - # Wait for engine logs - sleep 10s - echo 'Logs of Microgateway Engine container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer exit 1 } - create_sidecargateway() { - # create SidecarGateway resource for testing purposes - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done - kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - } - - trap clean_up EXIT - - echo "### Waiting for Microgateway Deployments to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ - deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then - fail 'Timout occurred' + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' fi - echo "" - echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" - # scale to zero replicas to ensure no pods are present from previous runs - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s - echo "" - - echo "### Waiting for backend pod" - i=0 - while true; do - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then - break - elif [ $i -gt 3 ]; then - fail 'Pod not ready' - fi - sleep 2s - i=$((i+1)) - done - - echo "### Checking Microgateway Engine sidecar container was injected" - if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then - fail 'Microgateway Engine sidecar container not injected' + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' fi - echo "True" - echo "" - echo "### Checking for valid license" - i=0 - while true; do - if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then - break - elif [ $i -gt 30 ]; then - fail 'Microgateway license is missing or invalid' - fi - sleep 2s - i=$((i+1)) - done - echo "True" - echo "" - - echo "### Create SidecarGateway resource for testing" - if ! create_sidecargateway ; then - fail 'Creation of SidecarGateway resource failed' + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' fi - echo "" - echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then - fail 'Timout occurred' - fi - echo "" + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac - echo "### Waiting for 'engine-config-valid' condition" - if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then - fail 'Configuration was never accepted by the Microgateway Engine' - fi - sleep 5s - echo "" - echo "" - - echo "### Checking whether a valid request is successful and returns HTTP status code '200'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "200 OK"; then - fail 'A valid request was not successful' - fi - echo "" - echo "" - - echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "400 Bad Request"; then - fail 'A malicious request was not blocked' - fi - echo "" - echo "" - - echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" - exit 0 - serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir {{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/values.schema.json b/charts/airlock/microgateway/4.2.3/values.schema.json index 208521ded..e087bd700 100644 --- a/charts/airlock/microgateway/4.2.3/values.schema.json +++ b/charts/airlock/microgateway/4.2.3/values.schema.json @@ -14,15 +14,6 @@ "commonAnnotations": { "$ref": "#/definitions/StringMap" }, - "crds": { - "type": "object", - "properties": { - "skipVersionCheck": { - "type": "boolean" - } - }, - "additionalProperties": false - }, "imagePullSecrets": { "type": "array", "items": { @@ -39,194 +30,120 @@ "additionalProperties": true } }, - "operator": { + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { "type": "object", "properties": { - "replicaCount": { - "type": "integer", - "minimum": 0 + "create": { + "type": "boolean" }, - "updateStrategy": { - "$ref": "#/definitions/UpdateStrategy" - }, - "image": { - "$ref": "#/definitions/Image" - }, - "podAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "podLabels": { - "$ref": "#/definitions/StringMap" - }, - "serviceAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "serviceLabels": { - "$ref": "#/definitions/StringMap" - }, - "resources": { - "type": "object" - }, - "nodeSelector": { - "$ref": "#/definitions/StringMap" - }, - "tolerations": { - "type": "array", - "items": { - "type": "object" - } - }, - "affinity": { - "type": "object" - }, - "config": { - "type": "object", - "properties": { - "logLevel": { - "type": "string", - "enum": [ - "debug", - "info", - "warn", - "error" - ] - } - }, - "required": [ - "logLevel" - ], - "additionalProperties": false - }, - "serviceAccount": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "annotations": { - "$ref": "#/definitions/StringMap" - }, - "name": { - "type": "string" - } - }, - "required": [ - "annotations", - "create", - "name" - ], - "additionalProperties": false - }, - "rbac": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - }, - "serviceMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false + "createSCCRole": { + "type": "boolean" } }, "required": [ - "affinity", - "config", - "image", - "updateStrategy", - "nodeSelector", - "podAnnotations", - "podLabels", - "rbac", - "replicaCount", - "resources", - "serviceAccount", - "serviceAnnotations", - "serviceLabels", - "serviceMonitor", - "tolerations" + "create", + "createSCCRole" ], "additionalProperties": false }, - "engine": { + "privileged": { + "type": "boolean" + }, + "serviceAccount": { "type": "object", "properties": { - "image": { - "$ref": "#/definitions/Image" + "create": { + "type": "boolean" }, - "resources": { - "type": "object" + "annotations": { + "$ref": "#/definitions/StringMap" }, - "sidecar": { - "type": "object", - "properties":{ - "podMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "required": [ - "podMonitor" - ], - "additionalProperties": false + "name": { + "type": "string" } }, "required": [ - "image", - "resources", - "sidecar" + "annotations", + "create", + "name" ], "additionalProperties": false }, - "networkValidator": { + "multusNetworkAttachmentDefinition": { "type": "object", "properties": { - "image": { - "$ref": "#/definitions/Image" + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" } }, "required": [ - "image" + "create", + "namespace" ], "additionalProperties": false }, - "license": { + "config": { "type": "object", "properties": { - "secretName": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { "type": "string", "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } } }, "required": [ - "secretName" + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" ], "additionalProperties": false }, @@ -247,16 +164,22 @@ } }, "required": [ + "affinity", "commonAnnotations", "commonLabels", - "crds", - "engine", + "config", "fullnameOverride", + "image", "imagePullSecrets", - "license", + "multusNetworkAttachmentDefinition", "nameOverride", - "operator", - "networkValidator", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", "tests" ], "additionalProperties": false, @@ -297,68 +220,6 @@ "tag" ], "additionalProperties": false - }, - "UpdateStrategy": { - "type": "object", - "oneOf" : [ - { - "properties": { - "type": { - "$ref": "#/definitions/RecreateType" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - }, - { - "properties": { - "type": { - "$ref": "#/definitions/RollingUpdateType" - }, - "rollingUpdate": { - "$ref": "#/definitions/RollingUpdate" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - } - ] - }, - "RecreateType": { - "type": "string", - "enum": [ - "Recreate" - ] - }, - "RollingUpdateType": { - "type": "string", - "enum": [ - "RollingUpdate" - ] - }, - "RollingUpdate": { - "type": "object", - "properties": { - "maxSurge": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - }, - "maxUnavailable": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - } - }, - "anyOf": [ - {"required": ["maxSurge"]}, - {"required": ["maxUnavailable"]} - ], - "additionalProperties": false } } } diff --git a/charts/airlock/microgateway/4.2.3/values.yaml b/charts/airlock/microgateway/4.2.3/values.yaml index 24772bb69..ba8acd686 100644 --- a/charts/airlock/microgateway/4.2.3/values.yaml +++ b/charts/airlock/microgateway/4.2.3/values.yaml @@ -1,4 +1,4 @@ -# -- Allows overriding the name to use instead of "microgateway". +# -- Allows overriding the name to use instead of "microgateway-cni". nameOverride: "" # -- Allows overriding the name to use as full name of resources. fullnameOverride: "" @@ -10,127 +10,71 @@ commonAnnotations: {} imagePullSecrets: [] # - name: myRegistryKeySecretName -crds: - # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. - # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster - # when performing a "helm install/upgrade". - skipVersionCheck: false -operator: - # -- Number of replicas for the operator Deployment. - replicaCount: 2 - # -- Specifies the operator update strategy. - updateStrategy: - type: RollingUpdate - # Specifies the Airlock Microgateway Operator image. - image: - # -- Image repository from which to pull the Airlock Microgateway Operator image. - repository: "quay.io/airlock/microgateway-operator" - # -- Image tag to pull. - tag: "4.2.3" - # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). - # Overrides tag when specified. - digest: "sha256:a429dfdb636e76bfbee7c59cfbe53d5f396c1f5603d5cb187f6283301ba4d7ba" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Annotations to add to all Pods. - podAnnotations: {} - # -- Labels to add to all Pods. - podLabels: {} - # -- Annotations to add to the Service. - serviceAnnotations: {} - # prometheus.io/scrape: "true" - # prometheus.io/port: "8080" - - # -- Labels to add to the Service. - serviceLabels: {} - # -- Resource restrictions to apply to the operator container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 1000m - # memory: 512Mi - # requests: - # cpu: 100m - # memory: 512Mi - - # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. - nodeSelector: {} - # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. - tolerations: [] - # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. - affinity: {} - # Parameters for the operator configuration. - config: - # -- Operator application log level. - logLevel: "info" - # Configures the generation of the ServiceAccount. - serviceAccount: - # -- Whether a ServiceAccount should be created. - create: true - # -- Annotations to add to the ServiceAccount. - annotations: {} - # -- Name of the ServiceAccount to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - # Configures the generation of Role and RoleBinding as well ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. - rbac: - # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. - create: true - # Configures the generation of a Prometheus Operator ServiceMonitor. - serviceMonitor: - # -- Whether to create a ServiceMonitor resource for monitoring. - create: false - # -- Labels to add to the ServiceMonitor. - labels: {} - # release: "" -engine: - # Specifies the Airlock Microgateway Engine image. - image: - # -- Image repository from which to pull the Airlock Microgateway Engine image. - repository: "quay.io/airlock/microgateway-engine" - # -- Image tag to pull. - tag: "4.2.3" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:9b0debeef611172aa5ca79c6b8cd045e56a3c883763ec62c0fa211bb86d35304" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Engine container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 500m - # memory: 128Mi - # requests: - # cpu: 10m - # memory: 40Mi - - # Additional configuration when deployed as a sidecar. - sidecar: - # Configures the generation of a Prometheus Operator PodMonitor. - podMonitor: - # -- Whether to create a PodMonitor resource for monitoring. - create: false - # -- Labels to add to the PodMonitor. - labels: {} - # release: "" -networkValidator: - # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. - image: - # -- Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. - repository: "cgr.dev/chainguard/busybox" - # -- Image tag to pull. - tag: "" - # -- SHA256 image digest to pull (in the format "sha256:a212cef6665b2464a41307162fa96e9623aa45c3fa32c39d320eae8b730d81e0"). - # Overrides tag when specified. - digest: "sha256:a212cef6665b2464a41307162fa96e9623aa45c3fa32c39d320eae8b730d81e0" - # -- Pull policy for this image. - pullPolicy: IfNotPresent -license: - # -- Name of the secret containing the "microgateway-license.txt" key. - secretName: "airlock-microgateway-license" -# Check whether the installation of the Airlock Microgateway Helm Chart was successful. -# Requires a secret with a valid Airlock Microgateway license key already to be present. +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.2.3" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:82b5924866840f783cce2e9b4095b7710a0e1cbf555498e8723ca811ca916290" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. + create: false + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system tests: # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). # If set to false, `helm test` will not run any tests. diff --git a/charts/airlock/microgateway/4.3.0/.helmignore b/charts/airlock/microgateway/4.3.0/.helmignore index 101ff5ac5..8561d2892 100644 --- a/charts/airlock/microgateway/4.3.0/.helmignore +++ b/charts/airlock/microgateway/4.3.0/.helmignore @@ -21,8 +21,7 @@ .idea/ *.tmproj .vscode/ -# CRDs kustomization.yaml -/crds/kustomization.yaml + # Helm unit tests /tests /validation diff --git a/charts/airlock/microgateway/4.3.0/Chart.yaml b/charts/airlock/microgateway/4.3.0/Chart.yaml index 41659f600..f36cc383b 100644 --- a/charts/airlock/microgateway/4.3.0/Chart.yaml +++ b/charts/airlock/microgateway/4.3.0/Chart.yaml @@ -9,15 +9,15 @@ annotations: - name: Airlock Microgateway Forum url: https://forum.airlock.com/ catalog.cattle.io/certified: partner - catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/display-name: Airlock Microgateway CNI catalog.cattle.io/kube-version: '>=1.25.0-0' - catalog.cattle.io/release-name: microgateway - charts.openshift.io/name: Airlock Microgateway + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI apiVersion: v2 appVersion: 4.3.0 -description: A Helm chart for deploying the Airlock Microgateway +description: A Helm chart for deploying the Airlock Microgateway CNI plugin home: https://www.airlock.com/en/microgateway -icon: file://assets/icons/microgateway.svg +icon: file://assets/icons/microgateway-cni.svg keywords: - WAF - Web Application Firewall @@ -30,14 +30,13 @@ keywords: - Filtering - DevSecOps - shift left -- control plane -- Operator +- CNI kubeVersion: '>=1.25.0-0' maintainers: - email: support@airlock.com name: Airlock url: https://www.airlock.com/ -name: microgateway +name: microgateway-cni sources: - https://github.com/airlock/microgateway type: application diff --git a/charts/airlock/microgateway/4.3.0/README.md b/charts/airlock/microgateway/4.3.0/README.md index 15ea0012e..2fa8977de 100644 --- a/charts/airlock/microgateway/4.3.0/README.md +++ b/charts/airlock/microgateway/4.3.0/README.md @@ -1,4 +1,4 @@ -# Airlock Microgateway +# Airlock Microgateway CNI ![Version: 4.3.0](https://img.shields.io/badge/Version-4.3.0-informational?style=flat-square) ![AppVersion: 4.3.0](https://img.shields.io/badge/AppVersion-4.3.0-informational?style=flat-square) @@ -40,58 +40,43 @@ Check the official documentation at **[docs.airlock.com](https://docs.airlock.co The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. ## Prerequisites -* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) -* [Airlock Microgateway License](#obtain-airlock-microgateway-license) -* [cert-manager](https://cert-manager.io/) * [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) -In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. -For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. -### Obtain Airlock Microgateway License -1. Either request a community or premium license - * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) - * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) -2. Check your inbox and save the license file microgateway-license.txt locally. - -> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. -### Deploy cert-manager -```bash -helm repo add jetstack https://charts.jetstack.io -helm install cert-manager jetstack/cert-manager --version '1.15.1' -n cert-manager --create-namespace --set crds.enabled=true --wait -``` - -## Deploy Airlock Microgateway Operator - -> This guide assumes a microgateway-license.txt file is present in the working directory. - -1. Install CRDs and Operator. +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). ```bash - # Create namespace - kubectl create namespace airlock-microgateway-system - - # Install License - kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt - - # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) - helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.3.0' --wait + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.0/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.0/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). 2. (Recommended) You can verify the correctness of the installation with `helm test`. ```bash - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.0' - helm test airlock-microgateway -n airlock-microgateway-system --logs - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.0' + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' ``` -### Upgrading CRDs - -The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. -CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: -```bash -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.3.0 --server-side --force-conflicts -``` - -**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. ## Support @@ -104,61 +89,33 @@ For the community edition, check our **[Airlock community forum](https://forum.a | Key | Type | Default | Description | |-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | | commonAnnotations | object | `{}` | Annotations to add to all resources. | | commonLabels | object | `{}` | Labels to add to all resources. | -| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | -| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | -| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | -| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | -| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | -| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | -| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | -| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | -| engine.image.digest | string | `"sha256:f442143294f3138965c9fa2734cafd39ebebe8e289600332b12f8a59c23dd9ef"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | -| engine.image.tag | string | `"4.3.0"` | Image tag to pull. | -| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | -| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | -| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | | fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:cb165e34a1ab1a903a9f38b741a7d78946470a118640310a41d2af8153d6e409"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.3.0"` | Image tag to pull. | | imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | -| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | -| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | -| networkValidator.image.digest | string | `"sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"` | SHA256 image digest to pull (in the format "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"). Overrides tag when specified. | -| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| networkValidator.image.repository | string | `"cgr.dev/chainguard/busybox"` | Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. | -| networkValidator.image.tag | string | `""` | Image tag to pull. | -| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | -| operator.config.logLevel | string | `"info"` | Operator application log level. | -| operator.image.digest | string | `"sha256:dc6f0f9a11d0336c10f6b8a5c7f64d98ac91bd90c49aa1dc4fe7b68cfdea8217"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | -| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | -| operator.image.tag | string | `"4.3.0"` | Image tag to pull. | -| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | -| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | -| operator.podLabels | object | `{}` | Labels to add to all Pods. | -| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | -| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | -| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | -| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | -| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | -| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | -| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | -| operator.serviceLabels | object | `{}` | Labels to add to the Service. | -| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | -| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | -| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | -| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | -| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | -| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | -| sessionAgent.image.digest | string | `"sha256:579dfded99145f9c2c1491ff1aeccb08721d63239a8b7f61bb9f455e17e968b2"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | -| sessionAgent.image.tag | string | `"4.3.0"` | Image tag to pull. | -| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | | tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | ## License diff --git a/charts/airlock/microgateway/4.3.0/app-readme.md b/charts/airlock/microgateway/4.3.0/app-readme.md deleted file mode 100644 index e32cac025..000000000 --- a/charts/airlock/microgateway/4.3.0/app-readme.md +++ /dev/null @@ -1,28 +0,0 @@ -# Airlock Microgateway - -*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* - -## Features -* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. -* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction -* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication -* Content security filters for protecting against known attacks (OWASP Top 10) -* Access control to allow only authenticated users to access the protected services -* API security features like JSON parsing or OpenAPI specification enforcement - -For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. - -## Requirements -* [Airlock Microgateway CNI Helm Chart](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Also available as Rancher Chart) -* [Airlock Microgateway License](https://github.com/airlock/microgateway?tab=readme-ov-file#obtain-airlock-microgateway-license) (After obtaining the license install it according to the [documentation](https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-operator)) -* [cert-manager](https://cert-manager.io/docs/installation/) - -## Documentation and links - -Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. - -* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) -* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) -* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) -* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) -* [GitHub](https://github.com/airlock/microgateway) \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml deleted file mode 100644 index 5c6215c90..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml +++ /dev/null @@ -1,124 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: accesscontrols.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: AccessControl - listKind: AccessControlList - plural: accesscontrols - singular: accesscontrol - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControl specifies the options to perform access control with a Microgateway Engine container. - 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: Specifies how the Airlock Microgateway Engine performs access control. - properties: - policies: - description: Policies configures access control policies. - items: - properties: - authorization: - description: Authorization configures how requests are authorized. An empty object value {} disables authorization. - properties: - authentication: - description: Authentication specifies that clients need to be authenticated with the provided method. - properties: - oidc: - description: OIDC configures client authentication using OpenID Connect. - properties: - oidcRelyingPartyRef: - description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - oidcRelyingPartyRef - type: object - type: object - type: object - identityPropagation: - description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. - properties: - actions: - description: Actions specifies the propagation actions. - items: - properties: - identityPropagationRef: - description: IdentityPropagationRef selects an IdentityPropagation to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - identityPropagationRef - type: object - type: array - onFailure: - description: |- - OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: - _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. - enum: - - Pass - type: string - required: - - actions - - onFailure - type: object - required: - - authorization - type: object - maxItems: 1 - minItems: 1 - type: array - required: - - policies - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml deleted file mode 100644 index 07ba25df5..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: envoyclusters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyCluster - listKind: EnvoyClusterList - plural: envoyclusters - singular: envoycluster - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyCluster is an additional Envoy Cluster resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy cluster. - properties: - value: - description: Value defines the Envoy Cluster which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml deleted file mode 100644 index cc09fbbb1..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,185 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: envoyconfigurations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyConfiguration - listKind: EnvoyConfigurationList - plural: envoyconfigurations - singular: envoyconfiguration - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - EnvoyConfiguration is the Schema for the envoyconfigurations API - {{% notice warning %}} EnvoyConfiguration resources may contain sensitive information and thus RBAC permissions should be granted with care. {{% /notice %}} - 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: EnvoyConfigurationSpec defines the desired state of EnvoyConfiguration - properties: - envoyResources: - properties: - clusters: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - endpoints: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - extensions: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - listeners: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - routes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - runtimes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - scopedRoutes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - secrets: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - envoyResourcesRaw: - description: |- - EnvoyResourcesRaw defines the desired state for each resource type. The resources are stored as zstd compressed JSON bytes. - For debugging purposes, the resources can be inspected with the following command: `kubectl get envoyconfiguration -ojsonpath='{.spec.envoyResourcesRaw}' | base64 -d | zstd -d | jq` - format: byte - type: string - nodeID: - description: '**Deprecated:** This field is now ignored as NodeID is always derived from the resource name.' - type: string - type: object - status: - description: EnvoyConfigurationStatus defines the observed state of EnvoyConfiguration - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of EnvoyConfiguration condition. - type: string - required: - - status - - type - type: object - type: array - status: - type: string - xds: - properties: - resourceTypes: - additionalProperties: - description: XdsResourceTypeSyncStatus defines the sync status of xDS for a specific resource type - properties: - errorMessage: - description: ErrorMessage defines an optional message why the currently served resources of this resource type are rejected by the client. - type: string - resources: - additionalProperties: - description: XdsResourceStatus defines the status of xDS for a specific resource - properties: - version: - description: Version defines the version which is currently served for this resource. - type: string - required: - - version - type: object - description: Resources defines the resources which are currently served for this resource type. - type: object - status: - description: Status defines the current sync status of this resource type. - type: string - version: - description: Version defines the version which is currently served for this resource type. - type: string - required: - - resources - - status - - version - type: object - description: ResourceTypes defines the sync statuses for each resource type. - type: object - version: - description: Version defines the version of the underlying xDS snapshot. - type: integer - required: - - version - type: object - required: - - status - - xds - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml deleted file mode 100644 index d6eb787ab..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: envoyhttpfilters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyHTTPFilter - listKind: EnvoyHTTPFilterList - plural: envoyhttpfilters - singular: envoyhttpfilter - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyHTTPFilter is an additional Envoy HTTP Filter resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy HTTP filter. - properties: - value: - description: Value defines the HTTP filter which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml deleted file mode 100644 index 77f8991e6..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml +++ /dev/null @@ -1,88 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: graphqls.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: GraphQL - listKind: GraphQLList - plural: graphqls - singular: graphql - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: GraphQL contains the configuration for the GraphQL specification. - 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: Specification of the desired GraphQL specification. - properties: - settings: - description: Settings defines the settings to configure GraphQL. - properties: - allowIntrospection: - default: true - description: AllowIntrospection specifies if the introspection system is exposed. - type: boolean - allowMutations: - default: true - description: AllowMutations specifies if mutations are allowed. - type: boolean - schema: - description: Specifies the GraphQL schema. - properties: - source: - description: Source specifies the GraphQL schema to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'schema.graphql'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml deleted file mode 100644 index 8fd43dc3a..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml +++ /dev/null @@ -1,759 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: headerrewrites.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: HeaderRewrites - listKind: HeaderRewritesList - plural: headerrewrites - singular: headerrewrites - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. - properties: - request: - description: Request defines manipulations on upstream request headers. - properties: - add: - description: Add defines which request headers will be added before forwarding to the upstream. - properties: - custom: - description: |- - Custom allows configuring additional upstream request headers. - Add selected headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which request headers will be forwarded to the upstream. - This can either be allHeaders or matchingHeaders. - Default: matchingHeaders: {...} - properties: - allHeaders: - description: AllHeaders specifies that all request headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which request headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - standardHeaders: - default: true - description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which request headers will be removed before forwarding to the upstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - alternativeForwardedHeaders: - default: true - description: |- - AlternativeForwardedHeaders removes downstream request headers which could potentially - be abused to alter the upstream's view of the remote connection. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - response: - description: Response defines manipulations on upstream response headers. - properties: - add: - description: Add defines which response headers will be added before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - csp: - default: true - description: |- - CSP sets a content security policy which allows only same-origin requests except for images - if the 'Content-Security-Policy' header is not set by the upstream. - type: boolean - featurePolicy: - default: false - description: |- - FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features - if the 'Feature-Policy' header is not set by the upstream. - **Deprecated:** Use permissionsPolicy instead. - type: boolean - hsts: - default: true - description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. - type: boolean - hstsPreload: - default: false - description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. - type: boolean - permissionsPolicy: - default: true - description: |- - PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features - if the 'Permissions-Policy' header is not set by the upstream. - type: boolean - referrerPolicy: - default: true - description: |- - ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests - if the 'Referrer-Policy' header is not set by the upstream. - type: boolean - xContentTypeOptions: - default: true - description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. - type: boolean - xFrameOptions: - default: true - description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which response headers will be forwarded to the downstream. - This can either be allHeaders or matchingHeaders. - Default: allHeaders: {} - properties: - allHeaders: - description: AllHeaders specifies that all response headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which response headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response header. - properties: - standardHeaders: - default: false - description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which response headers will be removed before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - auth: - description: Auth defines the categories of headers concerning authentication. - properties: - basic: - default: false - description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. - type: boolean - negotiate: - default: true - description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. - type: boolean - ntlm: - default: true - description: |- - NTLM removes upstream response headers that advise clients to authenticate with NTLM. - By default, these headers are removed, because NTLM pass-through is not supported. - type: boolean - type: object - informationLeakage: - description: InformationLeakage defines the categories of headers concerning information leakage. - properties: - application: - default: true - description: Application removes upstream response headers that leak information about the deployed software. - type: boolean - server: - default: true - description: Server removes upstream response headers that leak information about the server. - type: boolean - type: object - permissiveCors: - default: true - description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured remove operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - settings: - description: Settings configures the HeaderRewrites filter. - properties: - operationalMode: - default: Production - description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. - enum: - - Production - - Integration - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml deleted file mode 100644 index 4d37c5adb..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml +++ /dev/null @@ -1,358 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: parsers.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Parser - listKind: ParserList - plural: parsers - singular: parser - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Parser contains the configuration for content parsers (default and custom). - 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: Specification of the desired parser behavior. - properties: - request: - description: Request defines the parsing for downstream requests. - properties: - custom: - description: Custom allows configuring additional rules for parser selection. - properties: - rules: - description: |- - Rules defines a custom set prepended before built-in rules of enabled request parsers. - Disable all built-in parsers to overrule them completely. - items: - properties: - action: - description: |- - Action specifies what should happen when a request condition matches. - Only one of parse or skip can be set. - properties: - parse: - description: Parse activates the configured parser. - properties: - form: - description: Form activates the Form parser. - type: object - json: - description: JSON activates the JSON parser. - type: object - multipart: - description: Multipart activates the multipart parser. - type: object - type: object - skip: - description: Skip disables any content parsing - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - required: - - action - - requestConditions - type: object - type: array - type: object - defaultContentType: - default: application/x-www-form-urlencoded - description: DefaultContentType specifies the content-type header which should be injected into the request before parser selection if it is not already present and the request has a body. - minLength: 1 - type: string - parsers: - description: Parsers defines the configuration for the available content parsers. - properties: - form: - description: Form defines the configuration for the form parser. - properties: - enable: - default: true - description: Enable defines whether form payloads are inspected. - type: boolean - mediaTypePattern: - default: .*urlencoded.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as form arguments. - minLength: 1 - type: string - type: object - json: - description: JSON defines the configuration for the JSON parser. - properties: - enable: - default: true - description: Enable defines whether json payloads are inspected. - type: boolean - mediaTypePattern: - default: .*json.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as JSON. - minLength: 1 - type: string - type: object - multipart: - description: Multipart defines the configuration for the multipart parser. - properties: - enable: - default: true - description: Enable defines whether multipart payloads are inspected. - type: boolean - mediaTypePattern: - default: .*multipart.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as a multipart payload. - minLength: 1 - type: string - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml deleted file mode 100644 index ead724a75..000000000 --- a/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml +++ /dev/null @@ -1,758 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 - name: sidecargateways.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SidecarGateway - listKind: SidecarGatewayList - plural: sidecargateways - singular: sidecargateway - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: SidecarGateway contains the configuration how to configure the Airlock Microgateway Engine when used as Sidecar Container within the Pod of an application. - 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: Specification of the desired sidecar gateway behavior. - properties: - applications: - description: Applications defines applications which run on different ports. - items: - properties: - containerPort: - default: 8080 - description: |- - ContainerPort refers to the container port. - This must be a valid port number, 0 < x < 65536. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - downstream: - description: Downstream defines the downstream configuration for this application - properties: - protocol: - description: |- - Protocol defines the exposed HTTP protocol version. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies that the protocol should be inferred. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies that the client is assumed to speak HTTP/1.1. - type: object - http2: - description: HTTP2 specifies that the client is assumed to speak HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - remoteIP: - description: |- - RemoteIP defines how the remote IP of a client is propagated. - Default: xff: {...} - properties: - connectionIP: - description: ConnectionIP configures to use the source IP address of the direct downstream connection. - type: object - customHeader: - description: CustomHeader specifies to use a custom header for remote IP extraction. - properties: - headerName: - description: HeaderName specifies the name of the custom header containing the remote IP. - minLength: 1 - type: string - required: - default: true - description: Required specifies if the custom header is required. If true and not available the request will be rejected with 403. - type: boolean - required: - - headerName - type: object - xff: - description: XFF configures to use the standard 'X-Forwarded-For' header for IP extraction. - properties: - numTrustedHops: - default: 1 - description: NumTrustedHops specifies to extract the client's originating IP from the nth rightmost entry in the X-Forwarded-For header. With the default value of 1, the IP is extracted from the rightmost entry. - format: int32 - minimum: 1 - type: integer - type: object - type: object - requestNormalizations: - description: RequestNormalizations defines a set of normalization actions which are applied to the request before route matching. - properties: - mergeSlashes: - default: true - description: MergeSlashes ensures that adjacent slashes in the path are merged into one. - type: boolean - normalizePath: - default: true - description: NormalizePath ensures normalization according to RFC 3986 without case normalization. - type: boolean - type: object - restrictions: - description: Restrictions defines restrictions for downstream. - properties: - http: - description: HTTP defines limits for the HTTP protocol. - properties: - headersLength: - anyOf: - - type: integer - - type: string - default: 60Ki - description: HeadersLength defines maximum size of all request headers combined. Requests that exceed this limit will receive a 431 response. - 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 - timeouts: - description: Timeouts defines timeouts for downstream - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - default: 5m - description: |- - Idle defines the settings for the idle timeout when no data is sent or received. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - maxDuration: - default: 5m - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - requestHeaders: - default: 10s - description: |- - RequestHeaders defines the duration before all request headers must be received. - A value of 0 will completely disable the timeout. - Default: 10s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - clientCertificate: - description: |- - ClientCertificate defines the TLS settings for verification of client certificates. - At most one of ignored, optional and required can be set. - Default: ignored: {} - properties: - ignored: - description: Ignored disables verification of the client certificate. - type: object - optional: - description: |- - Optional enables verification of the client certificate if one is presented. - In this mode only trustedCA and crl settings can be configured since certificatePinning and allowedSANs require a client certificate. - properties: - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - required: - - trustedCA - type: object - required: - description: |- - Required contains settings for client certificate verification. A client must present a valid certificate. - At least one of trustedCA and certificatePinning must be set. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines the constraints a client certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - type: object - enable: - default: false - description: Enable defines if the downstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - secretRef: - description: SecretRef defines the reference to the TLS server certificate (secret of type kubernetes.io/tls). - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - xfcc: - description: |- - XFCC defines the handling of X-Forwarded-Client-Cert header. Meaning of the possible values: - _Sanitize_: Do not send the XFCC header to the next hop. This is the default value. - _ForwardOnly_: When the client connection is mTLS (Mutual TLS), forward the XFCC header in the request. - _AppendAndForward_: When the client connection is mTLS, append the client certificate information to the request’s XFCC header and forward it. - _SanitizeAndSet_: When the client connection is mTLS, reset the XFCC header with the client certificate information and send it to the next hop. - _AlwaysForwardOnly_: Always forward the XFCC header in the request, regardless of whether the client connection is mTLS. - Note: When forwarding the XFCC header in the request you might have to adjust the header length restrictions (See sidecargateway.spec.applications.downstream.restrictions.http) - enum: - - Sanitize - - ForwardOnly - - AppendAndForward - - SanitizeAndSet - - AlwaysForwardOnly - type: string - type: object - type: object - envoyHTTPFilterRefs: - description: EnvoyHTTPFilterRefs selects the relevant EnvoyHTTPFilters. - properties: - prepend: - description: Prepend selects the relevant EnvoyHTTPFilters which are added before those configured by the Airlock Microgateway. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - type: object - routes: - description: Routes defines the security configurations for different paths. The first matching route (from top to bottom) applies. - items: - description: |- - SidecarGatewayApplicationRoute defines the security configurations for different paths. - At most one of secured and unsecured can be set. - Default: secured: {...} - properties: - pathPrefix: - default: / - description: PathPrefix defines the path prefix used during route selection. - minLength: 1 - type: string - secured: - description: Secured enables WAF processing for this route. - properties: - accessControlRef: - description: |- - AccessControlRef selects the relevant AccessControl configuration resource. - If undefined, Airlock Microgateway does not perform any access control. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - contentSecurityRef: - description: |- - ContentSecurityRef selects the relevant ContentSecurity configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - unsecured: - description: |- - Unsecured disables all WAF functionality and therefore protection for this route. - WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. - type: object - type: object - type: array - x-kubernetes-list-map-keys: - - pathPrefix - x-kubernetes-list-type: map - telemetryRef: - description: |- - TelemetryRef selects the relevant Telemetry configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - upstream: - description: Upstream defines the upstream configuration for this application - properties: - protocol: - description: |- - Protocol defines HTTP protocol version used to communicate with the upstream. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies to negotiate the protocol with TLS ALPN (if TLS is enabled) or, as a fallback, use the same protocol that is used by the downstream connection. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies to use HTTP/1.1. - type: object - http2: - description: HTTP2 specifies to use HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - timeouts: - description: Timeouts defines the timeout settings. - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - description: |- - Timeout defines the settings for http timeouts. If this setting is not specified, the value of applications[].downstream.timeouts.http.idle is inherited. - A value of 0 will completely disable the timeout. - type: string - maxDuration: - default: 15s - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - Default: 15s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - enable: - default: false - description: Enable defines if the upstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - type: object - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - containerPort - x-kubernetes-list-type: map - envoyClusterRefs: - description: EnvoyClusterRefs selects the relevant EnvoyClusters. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - podSelector: - description: PodSelector defines to which Pods the configuration will be applied to. - properties: - matchLabels: - additionalProperties: - type: string - description: MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels. - type: object - type: object - sessionHandlingRef: - description: SessionHandlingRef selects the SessionHandling configuration to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - applications - type: object - status: - description: Most recently observed status of the SidecarGateway which is populated by the system. This data is read-only and may not be up to date. - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of SidecarGateway condition. - type: string - required: - - status - - type - type: object - type: array - pods: - items: - properties: - envoyConfig: - description: EnvoyConfig indicates the name of the EnvoyConfig CR for the Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - status: - type: string - unmanagedPods: - items: - properties: - managedBy: - description: ManagedBy indicates the Airlock Microgateway Operator instance which manages this Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - required: - - status - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.0/dashboards/license.json b/charts/airlock/microgateway/4.3.0/dashboards/license.json deleted file mode 100644 index b9d5777e2..000000000 --- a/charts/airlock/microgateway/4.3.0/dashboards/license.json +++ /dev/null @@ -1,521 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 0 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Expiry date of the Airlock Microgateway license associated with the selected operator.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "time: L" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 3, - "y": 0 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_expiry_timestamp_seconds{namespace=~\"${operator_namespace.regex}\"})*1000", - "instant": true, - "legendFormat": "Expiry Date (MM/DD/YYYY)", - "range": false, - "refId": "A" - } - ], - "title": "License Expiry Date", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of licensed requests for applications protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 7, - "y": 0 - }, - "id": 6, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_license_max_rq_count_per_month{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Licensed Requests", - "range": false, - "refId": "A" - } - ], - "title": "Licensed Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Estimated number of requests protected by Airlock Microgateway over 30 days based on the last 7 days.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 5, - "x": 11, - "y": 0 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d]))/7*30", - "instant": true, - "legendFormat": "Estimated Requests", - "range": false, - "refId": "A" - } - ], - "title": "Requests over 30 days (estimated)", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of requests per week processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 12, - "w": 16, - "x": 0, - "y": 4 - }, - "id": 5, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(avg_over_time(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d])[2m:30s]))", - "instant": false, - "legendFormat": "# Requests per week", - "range": true, - "refId": "A" - } - ], - "title": "Processed Requests per week", - "type": "timeseries" - } - ], - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "description": "", - "hide": 0, - "includeAll": false, - "label": "Operator Namespace", - "multi": false, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-7d", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway License", - "uid": "cdpq79bzrr01se", - "version": 2, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/dashboards/overview.json b/charts/airlock/microgateway/4.3.0/dashboards/overview.json deleted file mode 100644 index 094276621..000000000 --- a/charts/airlock/microgateway/4.3.0/dashboards/overview.json +++ /dev/null @@ -1,1138 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 3, - "title": "Overview", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of pods that are protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 1 - }, - "id": 11, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_sidecars{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Protected Pods", - "range": false, - "refId": "A" - } - ], - "title": "Protected Pods", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 3, - "y": 1 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [ - { - "options": { - "match": "nan", - "result": { - "index": 0, - "text": "n/a" - } - }, - "type": "special" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 6, - "y": 1 - }, - "id": 5, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", - "fullMetaSearch": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Blocked Requests (%)", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "% Blocked Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 9, - "y": 1 - }, - "id": 10, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License", - "type": "stat" - }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 5 - }, - "id": 2, - "title": "Blocks", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "left", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 0, - "y": 6 - }, - "id": 6, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "timezone": [ - "" - ], - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "instant": false, - "legendFormat": "Requests per second", - "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "hide": false, - "instant": false, - "legendFormat": "% Blocks", - "range": true, - "refId": "Blocks" - } - ], - "title": "Requests vs. % Blocks", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests blocked by Airlock Microgateway categorized by their corresponding type.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "custom": { - "align": "auto", - "cellOptions": { - "barAlignment": 0, - "drawStyle": "line", - "gradientMode": "none", - "hideValue": false, - "lineInterpolation": "linear", - "lineStyle": { - "dash": [ - 10, - 10 - ], - "fill": "solid" - }, - "showPoints": "never", - "spanNulls": false, - "type": "sparkline" - }, - "inspect": false - }, - "displayName": "Block Type", - "fieldMinMax": false, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "block_type" - }, - "properties": [ - { - "id": "custom.width", - "value": 153 - }, - { - "id": "custom.cellOptions", - "value": { - "type": "auto" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Trend #Block Types" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 12, - "y": 6 - }, - "id": 7, - "options": { - "cellHeight": "lg", - "footer": { - "countRows": false, - "enablePagination": false, - "fields": [ - "Value" - ], - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": false, - "sortBy": [ - { - "desc": true, - "displayName": "block_type" - } - ] - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m] offset -1m))/(60000/$__interval_ms)", - "format": "time_series", - "instant": false, - "legendFormat": "__auto", - "range": true, - "refId": "Block Types" - } - ], - "title": "Blocked Requests by Type", - "transformations": [ - { - "id": "timeSeriesTable", - "options": { - "A": { - "timeField": "Time" - }, - "Block Types": { - "stat": "sum", - "timeField": "Time" - } - } - } - ], - "type": "table" - }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 16 - }, - "id": 1, - "title": "Latency", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Percentiles of the application downstream latency over one minute.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "ms" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "25th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "50th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "95th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 17 - }, - "id": 8, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "25th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "50th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "95th Percentile" - } - ], - "title": "Application Downstream Latency", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Percentiles of the Airlock Microgateway processing time over one minute.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "ms" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "25th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "50th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "95th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 17 - }, - "id": 9, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "0.25 Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "0.5 Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "0.95 Percentile" - } - ], - "title": "Airlock Microgateway Processing Time", - "type": "timeseries" - } - ], - "refresh": "", - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "hide": 0, - "includeAll": true, - "label": "Operator Namespace", - "multi": true, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": ".*", - "skipUrlSync": false, - "sort": 0, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-24h", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway Overview", - "uid": "fdp5jb8fnrmyoa", - "version": 1, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/gke-values.yaml b/charts/airlock/microgateway/4.3.0/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway/4.3.0/openshift-values.yaml b/charts/airlock/microgateway/4.3.0/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway/4.3.0/questions.yml b/charts/airlock/microgateway/4.3.0/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway/4.3.0/templates/NOTES.txt b/charts/airlock/microgateway/4.3.0/templates/NOTES.txt index e38e3caa0..e8aa45888 100644 --- a/charts/airlock/microgateway/4.3.0/templates/NOTES.txt +++ b/charts/airlock/microgateway/4.3.0/templates/NOTES.txt @@ -1,34 +1,3 @@ -Thank you for installing Airlock Microgateway. -If you have not already done so, make sure that Airlock Microgateway CNI is also installed on the cluster. +Thank you for installing Airlock Microgateway CNI. -For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}. -Detailed CRD API reference documentation is also available at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}/api/crds. -{{ if .Values.crds.skipVersionCheck }} -- CRD version check skipped -{{- else }} -{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} -{{- if $outdatedCRDs -}} - {{- fail (printf ` - -Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. -Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: - -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts - -If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` - .Chart.AppVersion) - -}} -{{- end -}} -{{- end -}} -{{- if .Values.tests.enabled }} - {{- if .Values.operator.watchNamespaces -}} - {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) }} - {{- fail (printf ` - -To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. -` - .Release.Namespace) - -}} - {{- end -}} - {{- end -}} -{{- end }} \ No newline at end of file +For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" .}}. diff --git a/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl b/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl index 733ba9648..996491a87 100644 --- a/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl +++ b/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl @@ -1,16 +1,14 @@ {{/* Expand the name of the chart. -We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest explicit suffix is 14 characters. */}} -{{- define "airlock-microgateway.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Convert an image configuration object into an image ref string. */}} -{{- define "airlock-microgateway.image" -}} +{{- define "airlock-microgateway-cni.image" -}} {{- if .digest -}} {{- printf "%s@%s" .repository .digest -}} {{- else if .tag -}} @@ -22,19 +20,19 @@ Convert an image configuration object into an image ref string. {{/* Create a default fully qualified app name. -We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest implicit suffix is 27 characters. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. If release name contains chart name it will be used as a full name. */}} -{{- define "airlock-microgateway.fullname" -}} +{{- define "airlock-microgateway-cni.fullname" -}} {{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} {{- else }} {{- $name := default .Chart.Name .Values.nameOverride }} {{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} {{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} {{- end }} {{- end }} {{- end }} @@ -42,112 +40,62 @@ If release name contains chart name it will be used as a full name. {{/* Create chart name and version as used by the chart label. */}} -{{- define "airlock-microgateway.chart" -}} +{{- define "airlock-microgateway-cni.chart" -}} {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Common labels */}} -{{- define "airlock-microgateway.sharedLabels" -}} -helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} {{- if .Chart.AppVersion }} app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} {{- end }} app.kubernetes.io/managed-by: {{ .Release.Service }} -app.kubernetes.io/part-of: {{ .Chart.Name }} {{- with .Values.commonLabels }} {{ toYaml .}} {{- end }} {{- end }} {{/* -Common Selector labels +Common labels without component */}} -{{- define "airlock-microgateway.sharedSelectorLabels" -}} -app.kubernetes.io/instance: {{ .Release.Name }} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} {{- end }} {{/* -Restricted Container Security Context +Selector labels */}} -{{- define "airlock-microgateway.restrictedSecurityContext" -}} -allowPrivilegeEscalation: false -privileged: false -runAsNonRoot: true -capabilities: - drop: ["ALL"] -readOnlyRootFilesystem: true -seccompProfile: - type: RuntimeDefault +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} {{- end }} -{{/* Precondition: May only be used if AppVersion is isSemver */}} -{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} -{{- $version := (semver .Chart.AppVersion) -}} -{{- if $version.Prerelease -}} ->= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} -{{- else -}} ->= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 -{{- end -}} -{{- end -}} +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} -{{- define "airlock-microgateway.outdatedCRDs" -}} -{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} - {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} - {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} - {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} - {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} - {{- $isOutdated := false -}} - {{- if $crd -}} - {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} - {{- $isOutdated = true -}} - {{- if hasKey $crd.metadata "labels" -}} - {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} - {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} - {{- if (semverCompare $supportedVersion $crdVersion) }} - {{- $isOutdated = false -}} - {{- end }} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if $isOutdated }} -{{ base $path }} - {{- end }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{- define "airlock-microgateway.isSemver" -}} +{{- define "airlock-microgateway-cni.isSemver" -}} {{- regexMatch `^(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-]+)*))?$` . -}} {{- end -}} -{{- define "airlock-microgateway.docsVersion" -}} -{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} {{- $version := (semver .Chart.AppVersion) -}} {{- $version.Major }}.{{ $version.Minor -}} {{- else -}} {{- print "latest" -}} {{- end -}} {{- end -}} - -{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} -{{- $list := list -}} -{{- with .matchLabels -}} - {{- range $key, $value := . -}} - {{- $list = append $list (printf "%s=%s" $key $value) -}} - {{- end -}} -{{- end -}} -{{- with .matchExpressions -}} - {{- range . -}} - {{- if has .operator (list "In" "NotIn") -}} - {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} - {{- else if eq .operator "Exists" -}} - {{- $list = append $list .key -}} - {{- else if eq .operator "DoesNotExist" -}} - {{- $list = append $list (printf "!%s" .key) -}} - {{- end -}} - {{- end -}} -{{- end -}} -{{- join "," $list -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/clusterrole.yaml b/charts/airlock/microgateway/4.3.0/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/clusterrolebinding.yaml b/charts/airlock/microgateway/4.3.0/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/configmap.yaml b/charts/airlock/microgateway/4.3.0/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway/4.3.0/templates/daemonset.yaml b/charts/airlock/microgateway/4.3.0/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway/4.3.0/templates/network-attachment-definition.yaml b/charts/airlock/microgateway/4.3.0/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl deleted file mode 100644 index a540ff9f4..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl +++ /dev/null @@ -1,42 +0,0 @@ -{{/* -Create a default fully qualified name for operator components. -*/}} -{{- define "airlock-microgateway.operator.fullname" -}} -{{ include "airlock-microgateway.fullname" . }}-operator -{{- end }} - - -{{/* -Common operator labels -*/}} -{{- define "airlock-microgateway.operator.labels" -}} -{{ include "airlock-microgateway.sharedLabels" . }} -{{ include "airlock-microgateway.operator.selectorLabels" . }} -{{- end }} - -{{/* -Operator Selector labels -*/}} -{{- define "airlock-microgateway.operator.selectorLabels" -}} -{{ include "airlock-microgateway.sharedSelectorLabels" . }} -app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-operator -app.kubernetes.io/component: controller -{{- end }} - -{{/* -Create the name of the service account to use for the operator -*/}} -{{- define "airlock-microgateway.operator.serviceAccountName" -}} -{{- if .Values.operator.serviceAccount.create }} -{{- default (include "airlock-microgateway.operator.fullname" .) .Values.operator.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.operator.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -ServiceMonitor metrics regex pattern for leader only metrics -*/}} -{{- define "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" -}} -^(microgateway_license|microgateway_sidecars).*$ -{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl deleted file mode 100644 index 83b314cbc..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl +++ /dev/null @@ -1,237 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator rbac permission rules -*/}} -{{- define "airlock-microgateway-operator.rbacRules" -}} -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - pods - verbs: - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - "" - resources: - - pods/finalizers - verbs: - - update -- apiGroups: - - "" - resources: - - pods/status - verbs: - - patch - - update -- apiGroups: - - "" - resources: - - secrets - verbs: - - create - - delete - - get - - list - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - accesscontrols - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - contentsecurities - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - denyrules - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyclusters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - envoyhttpfilters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - graphqls - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - headerrewrites - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - identitypropagations - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - limits - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcrelyingparties - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - openapis - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - parsers - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - redisproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sessionhandlings - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways - verbs: - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/finalizers - verbs: - - update -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - telemetries - verbs: - - get - - list - - watch -{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl deleted file mode 100644 index 02e304890..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl +++ /dev/null @@ -1,339 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator mutating webhooks -*/}} -{{- define "airlock-microgateway-operator.mutatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /mutate-v1-pod - failurePolicy: Fail - name: mutate-pod.microgateway.airlock.com - reinvocationPolicy: IfNeeded - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -{{- end }} - -{{/* -Operator validating webhooks -*/}} -{{- define "airlock-microgateway-operator.validatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-v1-pod - failurePolicy: Fail - name: validate-pod.microgateway.airlock.com - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - - UPDATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-accesscontrol - failurePolicy: Fail - name: validate-accesscontrol.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - accesscontrols - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-denyrules - failurePolicy: Fail - name: validate-denyrules.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - denyrules - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoycluster - failurePolicy: Fail - name: validate-envoycluster.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyclusters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoyhttpfilter - failurePolicy: Fail - name: validate-envoyhttpfilter.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyhttpfilters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-graphql - failurePolicy: Fail - name: validate-graphql.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - graphqls - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-headerrewrites - failurePolicy: Fail - name: validate-headerrewrites.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - headerrewrites - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-identitypropagation - failurePolicy: Fail - name: validate-identitypropagation.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - identitypropagations - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-limits - failurePolicy: Fail - name: validate-limits.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - limits - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcprovider - failurePolicy: Fail - name: validate-oidcprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcrelyingparty - failurePolicy: Fail - name: validate-oidcrelyingparty.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcrelyingparties - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-openapi - failurePolicy: Fail - name: validate-openapi.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - openapis - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-parser - failurePolicy: Fail - name: validate-parser.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - parsers - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-redisprovider - failurePolicy: Fail - name: validate-redisprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - redisproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-sidecargateway - failurePolicy: Fail - name: validate-sidecargateway.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - sidecargateways - sideEffects: None -{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml deleted file mode 100644 index e86208023..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml +++ /dev/null @@ -1,394 +0,0 @@ -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -data: - engine_bootstrap_config_template.yaml: | - # Base configuration, admin interface on port 19000 - admin: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - dynamic_resources: - cds_config: - initial_fetch_timeout: 10s - resource_api_version: V3 - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - lds_config: - resource_api_version: V3 - initial_fetch_timeout: 10s - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - static_resources: - listeners: - - name: probe - address: - socket_address: - address: 0.0.0.0 - port_value: 19001 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: probe - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: probe - virtual_hosts: - - name: probe - domains: - - '*' - routes: - - name: ready - match: - path: /ready - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - - name: metrics - address: - socket_address: - address: 0.0.0.0 - port_value: 19002 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: metrics - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: metrics - virtual_hosts: - - name: metrics - domains: - - '*' - routes: - - name: metrics - match: - path: /metrics - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - prefix_rewrite: '/stats/prometheus' - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - clusters: - - name: xds_cluster - connect_timeout: 1s - type: STRICT_DNS - load_assignment: - cluster_name: xds_cluster - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: airlock-microgateway-operator-xds.$(OPERATOR_NAMESPACE).svc.cluster.local - port_value: 13377 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - transport_socket: - name: envoy.transport_sockets.tls - typed_config: - '@type': type.googleapis.com/envoy.extensions.transport_sockets.tls.v3.UpstreamTlsContext - common_tls_context: - tls_params: - tls_minimum_protocol_version: TLSv1_3 - tls_maximum_protocol_version: TLSv1_3 - validation_context_sds_secret_config: - name: validation_context_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/validation_context_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - tls_certificate_sds_secret_configs: - - name: tls_certificate_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/tls_certificate_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - - name: airlock_microgateway_engine_admin - connect_timeout: 1s - type: STATIC - load_assignment: - cluster_name: airlock_microgateway_engine_admin - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - stats_config: - stats_tags: - - tag_name: "block_type" - regex: "\\.(block_type\\.([^.]+))" - - tag_name: "attack_type" - regex: "\\.(attack_type\\.([^.]+))" - - tag_name: "envoy_cluster_name" - regex: "\\.(cluster\\.([^.]+))" - - tag_name: "version" - regex: "\\.(version\\.([^.]+))" - use_all_default_tags: true - overload_manager: - resource_monitors: - - name: "envoy.resource_monitors.global_downstream_max_connections" - typed_config: - "@type": type.googleapis.com/envoy.extensions.resource_monitors.downstream_connections.v3.DownstreamConnectionsConfig - max_active_downstream_connections: 50000 - bootstrap_extensions: - - name: airlock.bootstrap.engine_build_info - typed_config: - '@type': type.googleapis.com/airlock.extensions.bootstrap.stats.v1alpha.Stats - application_log_config: - log_format: - text_format: '{"@timestamp":"%Y-%m-%dT%T.%e%z","log":{"logger":"%n","level":"%l","origin":{"file":{"name":"%g","line":%#},"function":"%!"}},"event":{"module":"envoy","dataset":"envoy.application"},"process":{"pid":%P,"thread":{"id":%t}},"ecs":{"version":"8.5"},"message":"%j"}' - engine_container_template.yaml: | - name: "$(ENGINE_NAME)" - image: "$(ENGINE_IMAGE)" - imagePullPolicy: {{ .Values.engine.image.pullPolicy }} - args: - - "--config-path" - - "/etc/envoy/bootstrap_config.yaml" - - "--base-id" - - "$(BASE_ID)" - - "--file-flush-interval-msec" - - '1000' - - "--drain-time-s" - - '60' - - "--service-node" - - "$(POD_NAME).$(POD_NAMESPACE)" - - "--service-cluster" - - "$(APP_NAME).$(POD_NAMESPACE)" - - "--log-path" - - "/dev/stdout" - - "--log-level" - - "$(LOG_LEVEL)" - volumeMounts: - - name: airlock-microgateway-bootstrap-secret-volume - mountPath: /etc/envoy - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - - name: POD_IP - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: status.podIP - ports: - - containerPort: 13378 - protocol: TCP - - containerPort: 19001 - protocol: TCP - - containerPort: 19002 - protocol: TCP - livenessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 1 - readinessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.engine.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - session_agent_container_template.yaml: | - name: "$(SESSION_AGENT_NAME)" - image: "$(SESSION_AGENT_IMAGE)" - imagePullPolicy: {{ .Values.sessionAgent.image.pullPolicy }} - args: - - "--port" - - "19004" - - "--config-path" - - "/etc/microgateway-session-agent/config.json" - volumeMounts: - - name: airlock-microgateway-session-agent-volume - mountPath: /etc/microgateway-session-agent - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - ports: - - containerPort: 19004 - livenessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 5 - readinessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 5 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.sessionAgent.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - network_validator_container_template.yaml: | - name: "$(NETWORK_VALIDATOR_NAME)" - image: "$(NETWORK_VALIDATOR_IMAGE)" - imagePullPolicy: {{ .Values.networkValidator.image.pullPolicy }} - command: ["/bin/sh", "-c"] - args: - - |- - echo 'pong' | nc -v -l 127.0.0.1 -p 13378 & - for i in 1 2 3; do - sleep 1s - if r=$(echo 'ping' | nc 127.0.0.1 19003) && [ $r == pong ]; then - echo -n 'Traffic redirection to Airlock Microgateway Engine is working.' > /dev/termination-log - exit 0 - fi - done - echo -en 'Traffic redirection to Airlock Microgateway Engine is not working.\nRestart the pod after ensuring that hostNetwork is disabled and a compatible Airlock Microgateway CNI version is installed on the node.\nCertain environments may also require additional configuration (see docs.airlock.com for more information).' > /dev/termination-log - exit 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - operator_config.yaml: | - apiVersion: config.airlock.com/v1alpha1 - kind: OperatorConfig - health: - healthProbeBindAddress: :8081 - metrics: - bindAddress: 0.0.0.0:8080 - webhook: - port: 9443 - deployment: - sidecar: - engineContainerTemplate: "/sidecar/engine_container_template.yaml" - networkValidatorContainerTemplate: "/sidecar/network_validator_container_template.yaml" - sessionAgentContainerTemplate: "/sidecar/session_agent_container_template.yaml" - engine: - bootstrapConfigTemplate: "/engine_bootstrap_config_template.yaml" - log: - level: {{ .Values.operator.config.logLevel }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaces: - selector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaces: - list: - {{- toYaml . | nindent 8 }} - {{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml deleted file mode 100644 index 34d23f6d6..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml +++ /dev/null @@ -1,47 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} ---- -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-leader-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - operator.microgateway.airlock.com/isLeader: "true" - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml deleted file mode 100644 index 5378be8ef..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml deleted file mode 100644 index bafec1015..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml deleted file mode 100644 index 466c56338..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml +++ /dev/null @@ -1,13 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selfSigned: {} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml deleted file mode 100644 index 434d7e9d3..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml +++ /dev/null @@ -1,13 +0,0 @@ -{{- if .Values.operator.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with mustMerge .Values.operator.serviceAccount.annotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml deleted file mode 100644 index ff85a9a31..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml +++ /dev/null @@ -1,60 +0,0 @@ -{{- if .Values.operator.serviceMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - matchExpressions: - - { key: "operator.microgateway.airlock.com/isLeader", operator: DoesNotExist } - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: drop ---- -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - operator.microgateway.airlock.com/isLeader: "true" - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: keep -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml deleted file mode 100644 index 60b92e1e2..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - dnsNames: - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc.cluster.local - issuerRef: - kind: Issuer - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml deleted file mode 100644 index 477ea839f..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-webhook - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: https - name: webhook - port: 443 - protocol: TCP - targetPort: 9443 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml deleted file mode 100644 index 81b41acf5..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml +++ /dev/null @@ -1,24 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-xds - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: grpc - name: xds - port: 13377 - protocol: TCP - targetPort: 13377 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" diff --git a/charts/airlock/microgateway/4.3.0/templates/scc-role.yaml b/charts/airlock/microgateway/4.3.0/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/scc-rolebinding.yaml b/charts/airlock/microgateway/4.3.0/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/serviceaccount.yaml b/charts/airlock/microgateway/4.3.0/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml index 93bd4cd1b..744799333 100644 --- a/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml +++ b/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml @@ -2,142 +2,63 @@ apiVersion: v1 kind: ServiceAccount metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests --- apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests roleRef: apiGroup: rbac.authorization.k8s.io kind: Role - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" subjects: - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" --- apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests rules: - apiGroups: - - microgateway.airlock.com + - "apps" resources: - - sidecargateways + - daemonsets resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + - {{ include "airlock-microgateway-cni.fullname" . }} verbs: - - get - - list - - watch - - delete + - get + - watch + - list - apiGroups: - - microgateway.airlock.com + - "" resources: - - sidecargateways + - pods + - pods/log verbs: - - create + - get + - list +{{- if .Values.rbac.createSCCRole }} - apiGroups: - - "" - resources: - - events - verbs: - - list -- apiGroups: - - "apps" - resources: - - deployments + - security.openshift.io resourceNames: - - "{{ include "airlock-microgateway.operator.fullname" . }}" - verbs: - - get - - list - - watch -- apiGroups: - - "apps" + - privileged resources: - - statefulsets - - statefulsets/scale - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + - securitycontextconstraints verbs: - - get - - list - - watch - - patch -- apiGroups: - - "" - resources: - - pods - - pods/log - - pods/status - - pods/attach - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" - - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" - - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" - verbs: - - get - - list - - create - - watch - - delete -- apiGroups: - - "" - resources: - - pods - verbs: - - create -{{- if .Values.operator.watchNamespaceSelector }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -subjects: - - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -rules: -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list -{{- end }} + - use +{{- end -}} {{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml deleted file mode 100644 index 30ddc278d..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: v1 -kind: Service -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-service" - namespace: {{ .Release.Namespace }} - labels: - app: test-service - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - selector: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} - ports: - - name: http - port: 8080 - targetPort: 8080 -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml deleted file mode 100644 index 710a7b9f6..000000000 --- a/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: apps/v1 -kind: StatefulSet -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - serviceName: nginx - replicas: 0 - selector: - matchLabels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - k8s.v1.cni.cncf.io/networks: default/airlock-microgateway-cni - labels: - sidecar.microgateway.airlock.com/inject: "true" - sidecar.istio.io/inject: "false" - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 8 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 8 }} - spec: - containers: - - image: cgr.dev/chainguard/nginx - name: nginx - ports: - - containerPort: 8080 - volumeMounts: - - mountPath: /var/lib/nginx/tmp/ - name: nginx-tmp - - mountPath: /var/run - name: nginx-run - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 12 }} - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - emptyDir: {} - name: nginx-tmp - - emptyDir: {} - name: nginx-run -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml index ab82abea7..12d8c8de7 100644 --- a/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml +++ b/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml @@ -2,14 +2,11 @@ apiVersion: v1 kind: Pod metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" namespace: {{ .Release.Namespace }} labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - sidecar.istio.io/inject: "false" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} annotations: helm.sh/hook: test helm.sh/hook-delete-policy: before-hook-creation @@ -19,209 +16,88 @@ spec: - name: test image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true command: - sh - -c - | set -eu - clean_up() { - echo "" - echo "### Clean up test resources" - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - echo "" - echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s - sleep 3s - echo "" - } - fail() { + echo "Error: ${1}" echo "" - echo "### Error: ${1}" - echo "" - - if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then - echo "" - echo 'Microgateway Sidecargateway status:' - kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true - echo "" - echo "" - fi - - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then - echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" - kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true - echo "" - echo "" - echo 'Logs of Nginx container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true - echo "" - echo "" - # Wait for engine logs - sleep 10s - echo 'Logs of Microgateway Engine container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true - fi - + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer exit 1 } - create_sidecargateway() { - # create SidecarGateway resource for testing purposes - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done - kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - } - - {{- if .Values.operator.watchNamespaceSelector }} - echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" - if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then - labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') - fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` - .Release.Namespace - (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) - }} + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' fi - echo "" - {{- end }} - trap clean_up EXIT - echo "" - - echo "### Waiting for Microgateway Operator Deployments to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ - deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then - fail 'Timout occurred' + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' fi - echo "" - echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" - # scale to zero replicas to ensure no pods are present from previous runs - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s - echo "" - - echo "### Waiting for backend pod" - i=0 - while true; do - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then - break - elif [ $i -gt 3 ]; then - fail 'Pod not ready' - fi - sleep 2s - i=$((i+1)) - done - - echo "### Checking Microgateway Engine sidecar container was injected" - if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then - fail 'Microgateway Engine sidecar container not injected' + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' fi - echo "True" - echo "" - echo "### Checking for valid license" - i=0 - while true; do - if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then - break - elif [ $i -gt 30 ]; then - fail 'Microgateway license is missing or invalid' - fi - sleep 2s - i=$((i+1)) - done - echo "True" - echo "" + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac - echo "### Create SidecarGateway resource for testing" - if ! create_sidecargateway ; then - fail 'Creation of SidecarGateway resource failed' - fi - echo "" - - echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then - fail 'Timout occurred' - fi - echo "" - - echo "### Waiting for 'engine-config-valid' condition" - if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then - fail 'Configuration was never accepted by the Microgateway Engine' - fi - sleep 5s - echo "" - echo "" - - echo "### Checking whether a valid request is successful and returns HTTP status code '200'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "200 OK"; then - fail 'A valid request was not successful' - fi - echo "" - echo "" - - echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "400 Bad Request"; then - fail 'A malicious request was not blocked' - fi - echo "" - echo "" - - echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" - exit 0 - serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir {{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/values.schema.json b/charts/airlock/microgateway/4.3.0/values.schema.json index 173d6b084..e087bd700 100644 --- a/charts/airlock/microgateway/4.3.0/values.schema.json +++ b/charts/airlock/microgateway/4.3.0/values.schema.json @@ -14,15 +14,6 @@ "commonAnnotations": { "$ref": "#/definitions/StringMap" }, - "crds": { - "type": "object", - "properties": { - "skipVersionCheck": { - "type": "boolean" - } - }, - "additionalProperties": false - }, "imagePullSecrets": { "type": "array", "items": { @@ -39,304 +30,120 @@ "additionalProperties": true } }, - "operator": { + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { "type": "object", "properties": { - "replicaCount": { - "type": "integer", - "minimum": 0 - }, - "updateStrategy": { - "$ref": "#/definitions/UpdateStrategy" - }, - "image": { - "$ref": "#/definitions/Image" - }, - "podAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "podLabels": { - "$ref": "#/definitions/StringMap" - }, - "serviceAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "serviceLabels": { - "$ref": "#/definitions/StringMap" - }, - "resources": { - "type": "object" - }, - "nodeSelector": { - "$ref": "#/definitions/StringMap" - }, - "tolerations": { - "type": "array", - "items": { - "type": "object" - } - }, - "affinity": { - "type": "object" - }, - "config": { - "type": "object", - "properties": { - "logLevel": { - "type": "string", - "enum": [ - "debug", - "info", - "warn", - "error" - ] - } - }, - "required": [ - "logLevel" - ], - "additionalProperties": false - }, - "serviceAccount": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "annotations": { - "$ref": "#/definitions/StringMap" - }, - "name": { - "type": "string" - } - }, - "required": [ - "annotations", - "create", - "name" - ], - "additionalProperties": false - }, - "watchNamespaces": { - "type": "array", - "items": { - "type": "string" - } - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - }, - "rbac": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - }, - "serviceMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "oneOf": [ - { - "properties": { - "watchNamespaces": { - "minItems": 1 - }, - "watchNamespaceSelector": { - "additionalProperties": false - } - } - }, - { - "properties": { - "watchNamespaces": { - "maxItems": 0 - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - } - } - } - ], - "required": [ - "affinity", - "config", - "image", - "updateStrategy", - "nodeSelector", - "podAnnotations", - "podLabels", - "rbac", - "replicaCount", - "resources", - "serviceAccount", - "serviceAnnotations", - "serviceLabels", - "serviceMonitor", - "tolerations" - ], - "additionalProperties": false - }, - "engine": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - }, - "sidecar": { - "type": "object", - "properties":{ - "podMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "required": [ - "podMonitor" - ], - "additionalProperties": false - } - }, - "required": [ - "image", - "resources", - "sidecar" - ], - "additionalProperties": false - }, - "networkValidator": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - } - }, - "required": [ - "image" - ], - "additionalProperties": false - }, - "sessionAgent": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - } - }, - "required": [ - "image", - "resources" - ], - "additionalProperties": false - }, - "license": { - "type": "object", - "properties": { - "secretName": { - "type": "string", - "minLength": 1 - } - }, - "required": [ - "secretName" - ], - "additionalProperties": false - }, - "dashboards": { - "type": "object", - "properties" : { "create": { "type": "boolean" }, - "config": { - "type": "object", - "properties": { - "grafana": { - "type": "object", - "properties": { - "folderAnnotation": { - "$ref": "#/definitions/NameValuePair" - }, - "dashboardLabel": { - "$ref": "#/definitions/NameValuePair" - } - }, - "required": [ - "folderAnnotation", - "dashboardLabel" - ], - "additionalProperties": false - } - }, - "required": [ - "grafana" - ], - "additionalProperties": false - }, - "instances": { - "type": "object", - "properties": { - "overview": { - "$ref": "#/definitions/DashboardInstance" - }, - "license" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockMetrics" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockLogs" : { - "$ref": "#/definitions/DashboardInstance" - } - }, - "required": [ - "overview", - "license", - "blockMetrics", - "blockLogs" - ], - "additionalProperties": false + "createSCCRole": { + "type": "boolean" } }, "required": [ "create", - "config", - "instances" + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" ], "additionalProperties": false }, @@ -357,18 +164,22 @@ } }, "required": [ + "affinity", "commonAnnotations", "commonLabels", - "crds", - "engine", + "config", "fullnameOverride", + "image", "imagePullSecrets", - "license", + "multusNetworkAttachmentDefinition", "nameOverride", - "operator", - "networkValidator", - "sessionAgent", - "dashboards", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", "tests" ], "additionalProperties": false, @@ -409,132 +220,6 @@ "tag" ], "additionalProperties": false - }, - "LabelSelector": { - "type": "object", - "properties": { - "matchExpressions": { - "type": "array", - "items": { - "type": "object", - "required": [ - "key", - "operator" - ], - "properties": { - "key": { - "type": "string" - }, - "operator": { - "type": "string" - }, - "values": { - "type": "array", - "items": { - "type": "string" - } - } - }, - "additionalProperties": false - } - }, - "matchLabels": { - "$ref": "#/definitions/StringMap" - } - }, - "additionalProperties": false - }, - "UpdateStrategy": { - "type": "object", - "oneOf" : [ - { - "properties": { - "type": { - "$ref": "#/definitions/RecreateType" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - }, - { - "properties": { - "type": { - "$ref": "#/definitions/RollingUpdateType" - }, - "rollingUpdate": { - "$ref": "#/definitions/RollingUpdate" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - } - ] - }, - "RecreateType": { - "type": "string", - "enum": [ - "Recreate" - ] - }, - "RollingUpdateType": { - "type": "string", - "enum": [ - "RollingUpdate" - ] - }, - "RollingUpdate": { - "type": "object", - "properties": { - "maxSurge": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - }, - "maxUnavailable": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - } - }, - "anyOf": [ - {"required": ["maxSurge"]}, - {"required": ["maxUnavailable"]} - ], - "additionalProperties": false - }, - "DashboardInstance" : { - "type" : "object", - "properties" : { - "create" : { - "type" : "boolean" - } - }, - "required" : [ - "create" - ], - "additionalProperties": false - }, - "NameValuePair" : { - "type" : "object", - "properties" : { - "name" : { - "type": "string", - "minLength": 1 - }, - "value" : { - "type" : "string", - "minLength": 1 - } - }, - "required" : [ - "name", - "value" - ], - "additionalProperties": false } } } diff --git a/charts/airlock/microgateway/4.3.0/values.yaml b/charts/airlock/microgateway/4.3.0/values.yaml index b13232aa9..a5bf5dac5 100644 --- a/charts/airlock/microgateway/4.3.0/values.yaml +++ b/charts/airlock/microgateway/4.3.0/values.yaml @@ -1,4 +1,4 @@ -# -- Allows overriding the name to use instead of "microgateway". +# -- Allows overriding the name to use instead of "microgateway-cni". nameOverride: "" # -- Allows overriding the name to use as full name of resources. fullnameOverride: "" @@ -10,203 +10,75 @@ commonAnnotations: {} imagePullSecrets: [] # - name: myRegistryKeySecretName -crds: - # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. - # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster - # when performing a "helm install/upgrade". - skipVersionCheck: false -operator: - # -- Number of replicas for the operator Deployment. - replicaCount: 2 - # -- Specifies the operator update strategy. - updateStrategy: - type: RollingUpdate - # Specifies the Airlock Microgateway Operator image. - image: - # -- Image repository from which to pull the Airlock Microgateway Operator image. - repository: "quay.io/airlock/microgateway-operator" - # -- Image tag to pull. - tag: "4.3.0" - # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). - # Overrides tag when specified. - digest: "sha256:dc6f0f9a11d0336c10f6b8a5c7f64d98ac91bd90c49aa1dc4fe7b68cfdea8217" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Annotations to add to all Pods. - podAnnotations: {} - # -- Labels to add to all Pods. - podLabels: {} - # -- Annotations to add to the Service. - serviceAnnotations: {} - # prometheus.io/scrape: "true" - # prometheus.io/port: "8080" - - # -- Labels to add to the Service. - serviceLabels: {} - # -- Resource restrictions to apply to the operator container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 1000m - # memory: 512Mi - # requests: - # cpu: 100m - # memory: 512Mi - - # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. - nodeSelector: {} - # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. - tolerations: [] - # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. - affinity: {} - # Parameters for the operator configuration. - config: - # -- Operator application log level. - logLevel: "info" - # Configures the generation of the ServiceAccount. - serviceAccount: - # -- Whether a ServiceAccount should be created. - create: true - # -- Annotations to add to the ServiceAccount. - annotations: {} - # -- Name of the ServiceAccount to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - # -- Allows to restrict the operator to specific namespaces, depending on your needs. - # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). - # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. - # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. - # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. - # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. - # Please note that this feature requires a Premium license. - watchNamespaces: [] - # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. - # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. - # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). - # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. - # Please note that this feature requires a Premium license. - watchNamespaceSelector: {} - # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. - # matchLabels: - # microgateway.airlock.com/enable: "true" - # matchExpressions: - # - { key: environment, operator: NotIn, values: [dev] } - - # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. - rbac: - # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. - create: true - # Configures the generation of a Prometheus Operator ServiceMonitor. - serviceMonitor: - # -- Whether to create a ServiceMonitor resource for monitoring. - create: false - # -- Labels to add to the ServiceMonitor. - labels: {} - # release: "" -engine: - # Specifies the Airlock Microgateway Engine image. - image: - # -- Image repository from which to pull the Airlock Microgateway Engine image. - repository: "quay.io/airlock/microgateway-engine" - # -- Image tag to pull. - tag: "4.3.0" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:f442143294f3138965c9fa2734cafd39ebebe8e289600332b12f8a59c23dd9ef" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Engine container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 500m - # memory: 128Mi - # requests: - # cpu: 10m - # memory: 40Mi - - # Additional configuration when deployed as a sidecar. - sidecar: - # Configures the generation of a Prometheus Operator PodMonitor. - podMonitor: - # -- Whether to create a PodMonitor resource for monitoring. - create: false - # -- Labels to add to the PodMonitor. - labels: {} - # release: "" -networkValidator: - # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. - image: - # -- Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. - repository: "cgr.dev/chainguard/busybox" - # -- Image tag to pull. - tag: "" - # -- SHA256 image digest to pull (in the format "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"). - # Overrides tag when specified. - digest: "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6" - # -- Pull policy for this image. - pullPolicy: IfNotPresent -sessionAgent: - # Specifies the Airlock Microgateway Session Agent image. - image: - # -- Image repository from which to pull the Airlock Microgateway Session Agent image. - repository: "quay.io/airlock/microgateway-session-agent" - # -- Image tag to pull. - tag: "4.3.0" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:579dfded99145f9c2c1491ff1aeccb08721d63239a8b7f61bb9f455e17e968b2" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 150m - # memory: 32Mi - # requests: - # cpu: 10m - # memory: 8Mi -license: - # -- Name of the secret containing the "microgateway-license.txt" key. - secretName: "airlock-microgateway-license" -# Creates dashboards in the form of ConfigMaps that can be imported -# by Grafana using its sidecar setup. -dashboards: - # -- Whether to create any ConfigMaps containing Grafana dashboards to import. +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.3.0" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:cb165e34a1ab1a903a9f38b741a7d78946470a118640310a41d2af8153d6e409" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. create: false - config: - # Configures the necessary label and annotations along with their values - # to enable Grafana to correctly identify the ConfigMaps containing - # dashboards and file them within a dedicated folder in the dashboard overview. - # These settings need to match the Grafana sidecar configuration. - grafana: - folderAnnotation: - # -- Name of the annotation containing the folder name to file dashboards into. - name: "grafana_folder" - # -- Name of the folder dashboards are filed into within the Grafana UI. - value: "Airlock Microgateway" - dashboardLabel: - # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. - name: "grafana_dashboard" - # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. - value: "1" - instances: - # Available dashboard instances that can be individually created/deployed. - overview: - # -- Whether to create the overview dashboard. - create: true - license: - # -- Whether to create the license dashboard. - create: true - blockMetrics: - # -- Whether to create the block metrics dashboard. - create: true - blockLogs: - # -- Whether to create the block logs dashboard. - create: true -# Check whether the installation of the Airlock Microgateway Helm Chart was successful. -# Requires a secret with a valid Airlock Microgateway license key already to be present. + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system tests: # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). # If set to false, `helm test` will not run any tests. diff --git a/charts/airlock/microgateway/4.3.1/.helmignore b/charts/airlock/microgateway/4.3.1/.helmignore index 101ff5ac5..8561d2892 100644 --- a/charts/airlock/microgateway/4.3.1/.helmignore +++ b/charts/airlock/microgateway/4.3.1/.helmignore @@ -21,8 +21,7 @@ .idea/ *.tmproj .vscode/ -# CRDs kustomization.yaml -/crds/kustomization.yaml + # Helm unit tests /tests /validation diff --git a/charts/airlock/microgateway/4.3.1/Chart.yaml b/charts/airlock/microgateway/4.3.1/Chart.yaml index fefe5ea40..64a5a0387 100644 --- a/charts/airlock/microgateway/4.3.1/Chart.yaml +++ b/charts/airlock/microgateway/4.3.1/Chart.yaml @@ -9,15 +9,15 @@ annotations: - name: Airlock Microgateway Forum url: https://forum.airlock.com/ catalog.cattle.io/certified: partner - catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/display-name: Airlock Microgateway CNI catalog.cattle.io/kube-version: '>=1.25.0-0' - catalog.cattle.io/release-name: microgateway - charts.openshift.io/name: Airlock Microgateway + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI apiVersion: v2 appVersion: 4.3.1 -description: A Helm chart for deploying the Airlock Microgateway +description: A Helm chart for deploying the Airlock Microgateway CNI plugin home: https://www.airlock.com/en/microgateway -icon: file://assets/icons/microgateway.svg +icon: file://assets/icons/microgateway-cni.svg keywords: - WAF - Web Application Firewall @@ -30,14 +30,13 @@ keywords: - Filtering - DevSecOps - shift left -- control plane -- Operator +- CNI kubeVersion: '>=1.25.0-0' maintainers: - email: support@airlock.com name: Airlock url: https://www.airlock.com/ -name: microgateway +name: microgateway-cni sources: - https://github.com/airlock/microgateway type: application diff --git a/charts/airlock/microgateway/4.3.1/README.md b/charts/airlock/microgateway/4.3.1/README.md index f9a4c34e6..06e8e26ab 100644 --- a/charts/airlock/microgateway/4.3.1/README.md +++ b/charts/airlock/microgateway/4.3.1/README.md @@ -1,4 +1,4 @@ -# Airlock Microgateway +# Airlock Microgateway CNI ![Version: 4.3.1](https://img.shields.io/badge/Version-4.3.1-informational?style=flat-square) ![AppVersion: 4.3.1](https://img.shields.io/badge/AppVersion-4.3.1-informational?style=flat-square) @@ -40,58 +40,43 @@ Check the official documentation at **[docs.airlock.com](https://docs.airlock.co The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. ## Prerequisites -* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) -* [Airlock Microgateway License](#obtain-airlock-microgateway-license) -* [cert-manager](https://cert-manager.io/) * [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) -In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. -For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. -### Obtain Airlock Microgateway License -1. Either request a community or premium license - * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) - * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) -2. Check your inbox and save the license file microgateway-license.txt locally. - -> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. -### Deploy cert-manager -```bash -helm repo add jetstack https://charts.jetstack.io -helm install cert-manager jetstack/cert-manager --version '1.15.1' -n cert-manager --create-namespace --set crds.enabled=true --wait -``` - -## Deploy Airlock Microgateway Operator - -> This guide assumes a microgateway-license.txt file is present in the working directory. - -1. Install CRDs and Operator. +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). ```bash - # Create namespace - kubectl create namespace airlock-microgateway-system - - # Install License - kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt - - # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) - helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.3.1' --wait + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.1/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.1/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). 2. (Recommended) You can verify the correctness of the installation with `helm test`. ```bash - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.1' - helm test airlock-microgateway -n airlock-microgateway-system --logs - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.1' + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.1' ``` -### Upgrading CRDs - -The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. -CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: -```bash -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.3.1 --server-side --force-conflicts -``` - -**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. ## Support @@ -104,61 +89,33 @@ For the community edition, check our **[Airlock community forum](https://forum.a | Key | Type | Default | Description | |-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | | commonAnnotations | object | `{}` | Annotations to add to all resources. | | commonLabels | object | `{}` | Labels to add to all resources. | -| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | -| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | -| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | -| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | -| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | -| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | -| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | -| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | -| engine.image.digest | string | `"sha256:6be782cc3f3206bfa61f462812d2a495e114ae13c59a7cdaed7ca406d5bc1b01"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | -| engine.image.tag | string | `"4.3.1"` | Image tag to pull. | -| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | -| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | -| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | | fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:bdd216c8a8c56a0eee0134f67772cbd75358640a0685cf5d71add653abb2c53b"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.3.1"` | Image tag to pull. | | imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | -| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | -| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | -| networkValidator.image.digest | string | `"sha256:6626ab44066867687baa7bfcabedafce5adc50446be1207c90c3b211bd922f84"` | SHA256 image digest to pull (in the format "sha256:6626ab44066867687baa7bfcabedafce5adc50446be1207c90c3b211bd922f84"). Overrides tag when specified. | -| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| networkValidator.image.repository | string | `"cgr.dev/chainguard/netcat"` | Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. | -| networkValidator.image.tag | string | `""` | Image tag to pull. | -| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | -| operator.config.logLevel | string | `"info"` | Operator application log level. | -| operator.image.digest | string | `"sha256:84b6eb914103d4c62024d9f761b7dd4371ea3ba8996fb04095d87ebfaf3db2bb"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | -| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | -| operator.image.tag | string | `"4.3.1"` | Image tag to pull. | -| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | -| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | -| operator.podLabels | object | `{}` | Labels to add to all Pods. | -| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | -| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | -| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | -| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | -| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | -| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | -| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | -| operator.serviceLabels | object | `{}` | Labels to add to the Service. | -| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | -| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | -| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | -| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | -| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | -| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | -| sessionAgent.image.digest | string | `"sha256:d62bdb16c74d340a81791be1696d620950d8232437676910bb6e5548411f2afd"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | -| sessionAgent.image.tag | string | `"4.3.1"` | Image tag to pull. | -| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | | tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | ## License diff --git a/charts/airlock/microgateway/4.3.1/app-readme.md b/charts/airlock/microgateway/4.3.1/app-readme.md deleted file mode 100644 index e32cac025..000000000 --- a/charts/airlock/microgateway/4.3.1/app-readme.md +++ /dev/null @@ -1,28 +0,0 @@ -# Airlock Microgateway - -*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* - -## Features -* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. -* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction -* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication -* Content security filters for protecting against known attacks (OWASP Top 10) -* Access control to allow only authenticated users to access the protected services -* API security features like JSON parsing or OpenAPI specification enforcement - -For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. - -## Requirements -* [Airlock Microgateway CNI Helm Chart](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Also available as Rancher Chart) -* [Airlock Microgateway License](https://github.com/airlock/microgateway?tab=readme-ov-file#obtain-airlock-microgateway-license) (After obtaining the license install it according to the [documentation](https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-operator)) -* [cert-manager](https://cert-manager.io/docs/installation/) - -## Documentation and links - -Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. - -* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) -* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) -* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) -* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) -* [GitHub](https://github.com/airlock/microgateway) \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/accesscontrols.microgateway.airlock.com.yaml deleted file mode 100644 index bdca25f86..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/accesscontrols.microgateway.airlock.com.yaml +++ /dev/null @@ -1,124 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: accesscontrols.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: AccessControl - listKind: AccessControlList - plural: accesscontrols - singular: accesscontrol - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControl specifies the options to perform access control with a Microgateway Engine container. - 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: Specifies how the Airlock Microgateway Engine performs access control. - properties: - policies: - description: Policies configures access control policies. - items: - properties: - authorization: - description: Authorization configures how requests are authorized. An empty object value {} disables authorization. - properties: - authentication: - description: Authentication specifies that clients need to be authenticated with the provided method. - properties: - oidc: - description: OIDC configures client authentication using OpenID Connect. - properties: - oidcRelyingPartyRef: - description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - oidcRelyingPartyRef - type: object - type: object - type: object - identityPropagation: - description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. - properties: - actions: - description: Actions specifies the propagation actions. - items: - properties: - identityPropagationRef: - description: IdentityPropagationRef selects an IdentityPropagation to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - identityPropagationRef - type: object - type: array - onFailure: - description: |- - OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: - _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. - enum: - - Pass - type: string - required: - - actions - - onFailure - type: object - required: - - authorization - type: object - maxItems: 1 - minItems: 1 - type: array - required: - - policies - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/contentsecurities.microgateway.airlock.com.yaml deleted file mode 100644 index 0172657b5..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/contentsecurities.microgateway.airlock.com.yaml +++ /dev/null @@ -1,139 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: contentsecurities.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: ContentSecurity - listKind: ContentSecurityList - plural: contentsecurities - singular: contentsecurity - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: ContentSecurity specifies the options to secure an upstream web application with a Microgateway Engine container. - 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: Specifies the options to secure an upstream web application with a Microgateway Engine container. - properties: - apiProtection: - description: |- - APIProtection defines the relevant configurations to protect APIs. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - graphQLRef: - description: |- - GraphQLRef selects the relevant GraphQL configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - openAPIRef: - description: |- - OpenAPIRef selects the relevant OpenAPI configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - filter: - description: |- - Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests - to protect against various attack patterns. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - denyRulesRef: - description: |- - DenyRulesRef selects the relevant DenyRules configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - headerRewritesRef: - description: |- - HeaderRewritesRef selects the relevant HeaderRewrites. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - limitsRef: - description: |- - LimitsRef selects the relevant Limits configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - parserRef: - description: |- - ParserRef selects the relevant Parser configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/denyrules.microgateway.airlock.com.yaml deleted file mode 100644 index 6a73d53db..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/denyrules.microgateway.airlock.com.yaml +++ /dev/null @@ -1,1804 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: denyrules.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: DenyRules - listKind: DenyRulesList - plural: denyrules - singular: denyrules - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - DenyRules configures request filtering using Airlock built-in and custom deny rules. - Deny rules establish a negative security model. They define prohibited patterns which, when a match is found in a request, lead to it being blocked from reaching the upstream web application. - To handle possible false positives, lower the security level or define fine-granular deny rule exceptions - If undefined, default settings are applied, designed to work with most upstream web application services. - 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: Specification of the desired deny rules behavior. - properties: - request: - description: Request configures deny rules for downstream requests. - properties: - builtIn: - description: BuiltIn configures the built-in deny rules. - properties: - exceptions: - description: Exceptions allows to define exceptions for specific requests and deny rules. - items: - description: |- - DenyRulesException defines an exception for deny rules. Exceptions may be defined by any or a combination of the following elements: blockedData (the request data causing a block) or requestConditions (properties of a request without taking into consideration the reason why a request has been blocked). - At least one of blockedData and requestConditions must be set. - properties: - blockedData: - description: BlockedData defines an exception based on the request data causing the block. - properties: - graphQL: - description: |- - GraphQL defines an exception based on a blocked GraphQL query. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header defines an exception based on a blocked header. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON defines an exception based on a blocked JSON property. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - jsonPath: - description: |- - JSONPath defines the JSONPath pattern to match the path within the JSON. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - key: - description: |- - Key defines the key of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter defines an exception based on a blocked parameter. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source defines the source of the parameter. - enum: - - Query - - Post - - Any - type: string - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path defines an exception based on the blocked path. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment defines an exception based on a blocked path segment. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - segments: - description: Segments defines the position of a segment within the path. - properties: - index: - description: Index specifies an exact path segment position by index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value defines the value of a path segment. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - type: object - requestConditions: - description: RequestConditions defines an exception based on a property of a request without taking into consideration the reason why a request has been blocked. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKeys: - description: RuleKeys restricts the exception to a set of deny rules. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - type: object - type: array - overrides: - description: Overrides allows to override the builtIn settings for specific deny rules. - items: - description: DenyRulesOverride allows to override the builtIn settings for specific deny rules. - properties: - conditions: - description: Conditions select which built-in deny rules' settings will be adjusted. - properties: - ruleKeys: - description: RuleKeys is a list of built-in deny rule names. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - types: - description: Types defines the type of attributes the override should be applied on. If Types are defined without any RuleKeys the override is applied to all deny rules. - items: - description: |- - A deny rule override type name can be any of the following values: - Header | - Parameter | - Path | - JSON | - GraphQL - enum: - - Header - - Parameter - - Path - - PathSegment - - JSON - - GraphQL - type: string - minItems: 0 - type: array - type: object - settings: - description: Settings override the corresponding properties for the selected rules. - properties: - level: - description: Level specifies the filter strength. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: array - settings: - description: Settings contains the keys which will be adjusted. - properties: - level: - default: Standard - description: Level represents a set of deny rules with different filter strengths. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - type: object - type: object - custom: - description: Custom allows configuring additional deny rules. - properties: - rules: - description: Rules defines list of additional deny rules. - items: - properties: - blockData: - description: BlockData specifies the request data which should cause a block. - properties: - graphQL: - description: |- - GraphQL specifies to block requests containing a matching GraphQL property. - At least one of field, argument and value must be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header specifies to block requests containing a matching header. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON specifies to block requests containing a matching JSON property in the body. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - key: - description: Key defines the key of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter specifies to block requests containing a matching parameter. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path specifies to block requests with a matching path. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - matcher: - description: Matcher specifies which path to block. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment specifies to block requests containing a matching path segment. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - segments: - description: |- - Segments restricts which path segments are filtered by this rule. - If not specified, all segments of a path are filtered. - properties: - index: - description: Index restricts the rule to the path segment at this index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value specifies which path segment values to block. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - value - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKey: - description: RuleKey defines a technical key for the deny rule. Must be unique. - minLength: 1 - pattern: ^[A-Z][A-Z0-9_]*$ - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - required: - - blockData - - ruleKey - type: object - type: array - x-kubernetes-list-map-keys: - - ruleKey - x-kubernetes-list-type: map - type: object - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml deleted file mode 100644 index e0b7bb9fb..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: envoyhttpfilters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyHTTPFilter - listKind: EnvoyHTTPFilterList - plural: envoyhttpfilters - singular: envoyhttpfilter - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyHTTPFilter is an additional Envoy HTTP Filter resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy HTTP filter. - properties: - value: - description: Value defines the HTTP filter which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/graphqls.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/graphqls.microgateway.airlock.com.yaml deleted file mode 100644 index 43a8b6fc3..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/graphqls.microgateway.airlock.com.yaml +++ /dev/null @@ -1,88 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: graphqls.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: GraphQL - listKind: GraphQLList - plural: graphqls - singular: graphql - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: GraphQL contains the configuration for the GraphQL specification. - 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: Specification of the desired GraphQL specification. - properties: - settings: - description: Settings defines the settings to configure GraphQL. - properties: - allowIntrospection: - default: true - description: AllowIntrospection specifies if the introspection system is exposed. - type: boolean - allowMutations: - default: true - description: AllowMutations specifies if mutations are allowed. - type: boolean - schema: - description: Specifies the GraphQL schema. - properties: - source: - description: Source specifies the GraphQL schema to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'schema.graphql'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/headerrewrites.microgateway.airlock.com.yaml deleted file mode 100644 index 0916edd28..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/headerrewrites.microgateway.airlock.com.yaml +++ /dev/null @@ -1,759 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: headerrewrites.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: HeaderRewrites - listKind: HeaderRewritesList - plural: headerrewrites - singular: headerrewrites - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. - properties: - request: - description: Request defines manipulations on upstream request headers. - properties: - add: - description: Add defines which request headers will be added before forwarding to the upstream. - properties: - custom: - description: |- - Custom allows configuring additional upstream request headers. - Add selected headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which request headers will be forwarded to the upstream. - This can either be allHeaders or matchingHeaders. - Default: matchingHeaders: {...} - properties: - allHeaders: - description: AllHeaders specifies that all request headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which request headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - standardHeaders: - default: true - description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which request headers will be removed before forwarding to the upstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - alternativeForwardedHeaders: - default: true - description: |- - AlternativeForwardedHeaders removes downstream request headers which could potentially - be abused to alter the upstream's view of the remote connection. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - response: - description: Response defines manipulations on upstream response headers. - properties: - add: - description: Add defines which response headers will be added before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - csp: - default: true - description: |- - CSP sets a content security policy which allows only same-origin requests except for images - if the 'Content-Security-Policy' header is not set by the upstream. - type: boolean - featurePolicy: - default: false - description: |- - FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features - if the 'Feature-Policy' header is not set by the upstream. - **Deprecated:** Use permissionsPolicy instead. - type: boolean - hsts: - default: true - description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. - type: boolean - hstsPreload: - default: false - description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. - type: boolean - permissionsPolicy: - default: true - description: |- - PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features - if the 'Permissions-Policy' header is not set by the upstream. - type: boolean - referrerPolicy: - default: true - description: |- - ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests - if the 'Referrer-Policy' header is not set by the upstream. - type: boolean - xContentTypeOptions: - default: true - description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. - type: boolean - xFrameOptions: - default: true - description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which response headers will be forwarded to the downstream. - This can either be allHeaders or matchingHeaders. - Default: allHeaders: {} - properties: - allHeaders: - description: AllHeaders specifies that all response headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which response headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response header. - properties: - standardHeaders: - default: false - description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which response headers will be removed before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - auth: - description: Auth defines the categories of headers concerning authentication. - properties: - basic: - default: false - description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. - type: boolean - negotiate: - default: true - description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. - type: boolean - ntlm: - default: true - description: |- - NTLM removes upstream response headers that advise clients to authenticate with NTLM. - By default, these headers are removed, because NTLM pass-through is not supported. - type: boolean - type: object - informationLeakage: - description: InformationLeakage defines the categories of headers concerning information leakage. - properties: - application: - default: true - description: Application removes upstream response headers that leak information about the deployed software. - type: boolean - server: - default: true - description: Server removes upstream response headers that leak information about the server. - type: boolean - type: object - permissiveCors: - default: true - description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured remove operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - settings: - description: Settings configures the HeaderRewrites filter. - properties: - operationalMode: - default: Production - description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. - enum: - - Production - - Integration - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/identitypropagations.microgateway.airlock.com.yaml deleted file mode 100644 index 7cf5a5ce7..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/identitypropagations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,108 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: identitypropagations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: IdentityPropagation - listKind: IdentityPropagationList - plural: identitypropagations - singular: identitypropagation - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: IdentityPropagation specifies the desired identity propagation. - 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: Specification of the desired identity propagation. - properties: - header: - description: Header configures identity propagation via a request header. - properties: - name: - description: Name of the header to set. - minLength: 1 - type: string - value: - description: Value to propagate to the application. - properties: - source: - description: Source from which to extract the value. - properties: - metadata: - description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. - properties: - key: - description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. - minLength: 1 - type: string - namespace: - description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. - minLength: 1 - type: string - required: - - key - - namespace - type: object - oidc: - description: OIDC specifies to extract a value from the result of an OpenID Connect flow. - properties: - idToken: - description: IDToken specifies to extract the value from the OpenID Connect ID Token. - properties: - claim: - description: Claim selects the JWT claim from which to extract the value. - minLength: 1 - type: string - required: - - claim - type: object - required: - - idToken - type: object - type: object - required: - - source - type: object - required: - - name - - value - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/limits.microgateway.airlock.com.yaml deleted file mode 100644 index 894573dc5..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/limits.microgateway.airlock.com.yaml +++ /dev/null @@ -1,651 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: limits.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Limits - listKind: LimitsList - plural: limits - singular: limits - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Limits contains the configuration for limits. - 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: Specification of the desired limits behavior. - properties: - request: - description: Request defines the limits for requests. - properties: - limited: - description: Limited enables limits on request scope. - properties: - exceptions: - description: Exceptions defines limit exceptions. - items: - description: LimitsException defines an exception for limits. - properties: - length: - description: Length defines an exception for length limits based on the data element exceeding the limit. - properties: - graphQL: - description: GraphQL defines a field, argument or value length limit exception for a GraphQL query. - properties: - argument: - description: |- - Argument restricts the exception to GraphQL queries with a matching argument of a field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field restricts the exception to GraphQL queries with a matching field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value restricts the exception to GraphQL queries with a matching argument value. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: JSON defines a key and value length limit exception for a JSON property. - properties: - jsonPath: - description: |- - JSONPath restricts the exception to JSON properties with a matching JSONPath. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - required: - - jsonPath - type: object - parameter: - description: Parameter defines a name and value length limit exception for a parameter. - properties: - name: - description: Name restricts the exception to parameters with a matching name. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source restricts the exception to parameters of this kind. - enum: - - Query - - Post - - Any - type: string - required: - - name - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this exception to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - type: object - type: array - general: - description: General defines general request limits. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the request body. It specifies the number of bytes (0 = unlimited). This limit is effective for any request not processed by one of the content parsers (e.g. json) as configured in the Parser CRD. **Note** This limit does not apply to WebSocket or gRPC traffic. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - pathLength: - anyOf: - - type: integer - - type: string - default: 1Ki - description: PathLength defines the maximum path length for all requests (parsed and unparsed). - 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 - graphQL: - description: GraphQL defines the limits for GraphQL requests. - properties: - nestingDepth: - default: 10 - description: NestingDepth defines the maximum depth of nesting for GraphQL objects. - format: int64 - type: integer - querySize: - anyOf: - - type: integer - - type: string - default: 1Ki - description: QuerySize defines the maximum size for GraphQL queries. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: "256" - description: ValueLength defines the maximum length for GraphQL values. - 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 - json: - description: JSON defines the limits for JSON requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the JSON request body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - elementCount: - default: 10000 - description: ElementCount defines the maximum number of keys and array items in the whole JSON document (recursive). - format: int64 - type: integer - keyCount: - default: 250 - description: KeyCount defines the maximum number of keys of a single JSON object (non-recursive). - format: int64 - type: integer - keyLength: - anyOf: - - type: integer - - type: string - default: "128" - description: KeyLength defines the maximum length for JSON keys. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - nestingDepth: - default: 100 - description: NestingDepth defines the maximum depth of nesting for JSON objects and JSON arrays. - format: int64 - type: integer - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for JSON values. - 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 - multipart: - description: Multipart defines the limits for Multipart requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the Multipart request body. It specifies the number of bytes (0 = unlimited). - 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 - parameter: - description: Parameter defines the limits for request parameters. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the form data body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - count: - default: 128 - description: Count defines the maximum number of request parameters. - format: int64 - type: integer - nameLength: - anyOf: - - type: integer - - type: string - default: "128" - description: NameLength defines the maximum length for parameter names. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for parameter values. - 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 - unlimited: - description: Unlimited disables all limits on request scope. - type: object - type: object - settings: - description: Settings configures the limits filter. - properties: - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a limit hits. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/oidcproviders.microgateway.airlock.com.yaml deleted file mode 100644 index eba2dbf7c..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/oidcproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,305 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: oidcproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCProvider - listKind: OIDCProviderList - plural: oidcproviders - singular: oidcprovider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCProvider specifies an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - 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: Specification of an OpenID Provider. - properties: - static: - description: Static configures an OpenID Provider by explicitly specifying all endpoints. - properties: - endpoints: - description: Endpoints specifies the OpenID Provider endpoints. - properties: - authorization: - description: Authorization specifies the endpoint to which the authorization request is sent. - properties: - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - token: - description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. - properties: - tls: - description: TLS defines TLS settings. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: |- - Custom explicitly specifies how the server certificate should be verified. - Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines constraints the presented certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - disabled: - description: |- - Disabled specifies to trust any certificate without verification. - THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. - type: object - type: object - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - required: - - authorization - - token - type: object - required: - - endpoints - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml deleted file mode 100644 index 863f039ef..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml +++ /dev/null @@ -1,224 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: oidcrelyingparties.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCRelyingParty - listKind: OIDCRelyingPartyList - plural: oidcrelyingparties - singular: oidcrelyingparty - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} - 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: Specification of the OIDC Relying Party configuration. - properties: - clientID: - description: ClientID specifies the OIDCRelyingParty "client_id". - minLength: 1 - type: string - credentials: - description: Credentials used for client authentication on the back-channel with the authorization server. - properties: - clientSecret: - description: ClientSecret authenticates with the client password issued by the OpenID Provider (OP). - properties: - method: - default: BasicAuth - description: Method specifies in which format the client secret is sent with the authorization request. - enum: - - BasicAuth - - FormURLEncoded - type: string - secretRef: - description: SecretRef specifies the kubernetes secret containing the client password with key "client.secret". - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - clientSecret - type: object - oidcProviderRef: - description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - pathMapping: - description: PathMapping configures the action matching. - properties: - logoutPath: - description: LogoutPath specifies which request paths should initiate a logout. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - redirectPath: - description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - logoutPath - - redirectPath - type: object - redirectURI: - description: |- - RedirectURI configures the "redirect_uri" parameter included in the authorization request. - May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. - minLength: 1 - type: string - required: - - clientID - - credentials - - oidcProviderRef - - pathMapping - - redirectURI - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.1/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/openapis.microgateway.airlock.com.yaml deleted file mode 100644 index 5f16d2993..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/openapis.microgateway.airlock.com.yaml +++ /dev/null @@ -1,167 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: openapis.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OpenAPI - listKind: OpenAPIList - plural: openapis - singular: openapi - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: OpenAPI contains the configuration for the OpenAPI specification. - 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: Specification of the desired OpenAPI specification. - properties: - response: - description: Response defines the validation behaviour for responses. - properties: - secured: - description: Secured enables response checking. - properties: - validation: - default: Lax - description: Validation defines the validation mode for responses. - enum: - - Lax - - Strict - type: string - type: object - unsecured: - description: Unsecured disables response checking. - type: object - type: object - settings: - description: Settings defines the settings to configure OpenAPI specification enforcement. - properties: - logging: - description: Logging specifies the access log behavior. - properties: - maxFailedSubvalidations: - default: 10 - description: MaxFailedSubvalidations defines the maximum number of failed subvalidations being logged. - format: int64 - type: integer - type: object - schema: - description: Schema configures the OpenAPI specification. - properties: - source: - description: Source specifies the OpenAPI specification to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'openapi.json'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - validation: - description: Validation specifies the patterns for the validation behavior. - properties: - authentication: - description: Authentication defines the settings for the authentication scheme. - properties: - oAuth2: - description: OAuth2 specifies the OAuth2 parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - oidc: - description: Oidc specifies the OIDC parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - type: object - type: object - required: - - schema - type: object - required: - - settings - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/parsers.microgateway.airlock.com.yaml deleted file mode 100644 index 61e46f6cf..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/parsers.microgateway.airlock.com.yaml +++ /dev/null @@ -1,358 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: parsers.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Parser - listKind: ParserList - plural: parsers - singular: parser - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Parser contains the configuration for content parsers (default and custom). - 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: Specification of the desired parser behavior. - properties: - request: - description: Request defines the parsing for downstream requests. - properties: - custom: - description: Custom allows configuring additional rules for parser selection. - properties: - rules: - description: |- - Rules defines a custom set prepended before built-in rules of enabled request parsers. - Disable all built-in parsers to overrule them completely. - items: - properties: - action: - description: |- - Action specifies what should happen when a request condition matches. - Only one of parse or skip can be set. - properties: - parse: - description: Parse activates the configured parser. - properties: - form: - description: Form activates the Form parser. - type: object - json: - description: JSON activates the JSON parser. - type: object - multipart: - description: Multipart activates the multipart parser. - type: object - type: object - skip: - description: Skip disables any content parsing - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - required: - - action - - requestConditions - type: object - type: array - type: object - defaultContentType: - default: application/x-www-form-urlencoded - description: DefaultContentType specifies the content-type header which should be injected into the request before parser selection if it is not already present and the request has a body. - minLength: 1 - type: string - parsers: - description: Parsers defines the configuration for the available content parsers. - properties: - form: - description: Form defines the configuration for the form parser. - properties: - enable: - default: true - description: Enable defines whether form payloads are inspected. - type: boolean - mediaTypePattern: - default: .*urlencoded.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as form arguments. - minLength: 1 - type: string - type: object - json: - description: JSON defines the configuration for the JSON parser. - properties: - enable: - default: true - description: Enable defines whether json payloads are inspected. - type: boolean - mediaTypePattern: - default: .*json.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as JSON. - minLength: 1 - type: string - type: object - multipart: - description: Multipart defines the configuration for the multipart parser. - properties: - enable: - default: true - description: Enable defines whether multipart payloads are inspected. - type: boolean - mediaTypePattern: - default: .*multipart.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as a multipart payload. - minLength: 1 - type: string - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/redisproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/redisproviders.microgateway.airlock.com.yaml deleted file mode 100644 index d7d37a5ae..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/redisproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,159 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: redisproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: RedisProvider - listKind: RedisProviderList - plural: redisproviders - singular: redisprovider - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: RedisProvider contains a client configuration for connecting to a Redis database. - 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: Specification of a Redis database client configuration. - properties: - auth: - description: Auth specifies the Redis credentials. - properties: - password: - description: Password specifies the Redis password. - properties: - secretRef: - description: SecretRef selects the secret containing the Redis password under the key 'redis.password'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - username: - default: default - description: Username specifies the Redis username to authenticate with. - minLength: 1 - pattern: ^[^\s]+$ - type: string - required: - - password - type: object - mode: - description: Mode configures the redis deployment mode. - properties: - standalone: - description: Standalone specifies the standalone Redis instance to connect to. - properties: - host: - description: Host specifies the IP or hostname. - minLength: 1 - pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ - type: string - port: - default: 6379 - description: Port specifies the port. - maximum: 65535 - minimum: 1 - type: integer - required: - - host - type: object - type: object - timeouts: - description: Timeouts specifies the timeouts when interacting with the Redis endpoint. - properties: - connect: - default: 5s - description: Connect specifies the timeout for establishing a connection. - type: string - maxDuration: - default: 2s - description: MaxDuration specifies the response timeout. - type: string - type: object - tls: - description: TLS defines TLS settings. If not specified, TLS is disabled i.e. unencrypted TCP is used when connecting to the Redis instance. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: Custom explicitly specifies how the server certificate should be verified. - properties: - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - required: - - certificates - type: object - required: - - trustedCA - type: object - disabled: - description: 'Disabled specifies to trust any certificate without verification. THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. Note: This setting currently also disables TLS SNI.' - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching the host and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Session Agent’s base image. - type: object - type: object - type: object - required: - - mode - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/sessionhandlings.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/sessionhandlings.microgateway.airlock.com.yaml deleted file mode 100644 index c4b51ef6c..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/sessionhandlings.microgateway.airlock.com.yaml +++ /dev/null @@ -1,77 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: sessionhandlings.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SessionHandling - listKind: SessionHandlingList - plural: sessionhandlings - singular: sessionhandling - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - SessionHandling contains the configuration for session handling. - - - {{% notice warning %}} The Session Handling feature (required for OIDC) is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as high-availability Redis configurations (e.g. Sentinel/Cluster) are not yet supported. - {{% /notice %}} - 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: Specification of the desired session handling behavior. - properties: - persistence: - description: Persistence configures where to store the session state. - properties: - redisProviderRef: - description: RedisProviderRef specifies to cache session information in the provided Redis instance. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - redisProviderRef - type: object - required: - - persistence - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.1/crds/telemetries.microgateway.airlock.com.yaml deleted file mode 100644 index 8f911d3d2..000000000 --- a/charts/airlock/microgateway/4.3.1/crds/telemetries.microgateway.airlock.com.yaml +++ /dev/null @@ -1,96 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 - name: telemetries.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Telemetry - listKind: TelemetryList - plural: telemetries - singular: telemetry - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Telemetry contains the configuration for telemetry (logging, metrics & tracing). - 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: Specification of the desired telemetry behavior. - properties: - correlation: - description: Correlation defines the correlation aspects of Telemetry. - properties: - idSource: - description: IDSource specifies how an external correlation ID should be obtained for a request. If not specified, no correlation ID will be logged. - properties: - header: - description: Header specifies to extract the correlation ID from a request header. If the header is absent from a request, no correlation ID will be logged. - properties: - name: - default: X-Correlation-Id - description: Name of the header (case-insensitive) from which to extract the correlation ID. - minLength: 1 - type: string - type: object - required: - - header - type: object - request: - description: Request defines the request related correlation settings of Telemetry. - properties: - allowDownstreamRequestID: - default: true - description: AllowDownstreamRequestID defines whether trace sampling will consider a provided x-request-id. - type: boolean - alterRequestID: - default: true - description: AlterRequestID defines whether to alter the UUID to reflect the trace sampling decision. If disabled no modification to the UUID will be performed, this may break tracing in the upstream. - type: boolean - type: object - type: object - logging: - description: Logging defines the logging aspects of Telemetry. - properties: - accessLog: - description: AccessLog defines the access log settings of Telemetry. - properties: - format: - description: Format defines the Access Log format of the sidecar. - properties: - json: - description: JSON defines the Access Log format as JSON. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.1/dashboards/license.json b/charts/airlock/microgateway/4.3.1/dashboards/license.json deleted file mode 100644 index b9d5777e2..000000000 --- a/charts/airlock/microgateway/4.3.1/dashboards/license.json +++ /dev/null @@ -1,521 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 0 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Expiry date of the Airlock Microgateway license associated with the selected operator.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "time: L" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 3, - "y": 0 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_expiry_timestamp_seconds{namespace=~\"${operator_namespace.regex}\"})*1000", - "instant": true, - "legendFormat": "Expiry Date (MM/DD/YYYY)", - "range": false, - "refId": "A" - } - ], - "title": "License Expiry Date", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of licensed requests for applications protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 7, - "y": 0 - }, - "id": 6, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_license_max_rq_count_per_month{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Licensed Requests", - "range": false, - "refId": "A" - } - ], - "title": "Licensed Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Estimated number of requests protected by Airlock Microgateway over 30 days based on the last 7 days.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 5, - "x": 11, - "y": 0 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d]))/7*30", - "instant": true, - "legendFormat": "Estimated Requests", - "range": false, - "refId": "A" - } - ], - "title": "Requests over 30 days (estimated)", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of requests per week processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 12, - "w": 16, - "x": 0, - "y": 4 - }, - "id": 5, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(avg_over_time(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d])[2m:30s]))", - "instant": false, - "legendFormat": "# Requests per week", - "range": true, - "refId": "A" - } - ], - "title": "Processed Requests per week", - "type": "timeseries" - } - ], - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "description": "", - "hide": 0, - "includeAll": false, - "label": "Operator Namespace", - "multi": false, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-7d", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway License", - "uid": "cdpq79bzrr01se", - "version": 2, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/dashboards/overview.json b/charts/airlock/microgateway/4.3.1/dashboards/overview.json deleted file mode 100644 index 094276621..000000000 --- a/charts/airlock/microgateway/4.3.1/dashboards/overview.json +++ /dev/null @@ -1,1138 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 3, - "title": "Overview", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of pods that are protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 1 - }, - "id": 11, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_sidecars{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Protected Pods", - "range": false, - "refId": "A" - } - ], - "title": "Protected Pods", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 3, - "y": 1 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [ - { - "options": { - "match": "nan", - "result": { - "index": 0, - "text": "n/a" - } - }, - "type": "special" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 6, - "y": 1 - }, - "id": 5, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", - "fullMetaSearch": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Blocked Requests (%)", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "% Blocked Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 9, - "y": 1 - }, - "id": 10, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License", - "type": "stat" - }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 5 - }, - "id": 2, - "title": "Blocks", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "left", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 0, - "y": 6 - }, - "id": 6, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "timezone": [ - "" - ], - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "instant": false, - "legendFormat": "Requests per second", - "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "hide": false, - "instant": false, - "legendFormat": "% Blocks", - "range": true, - "refId": "Blocks" - } - ], - "title": "Requests vs. % Blocks", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests blocked by Airlock Microgateway categorized by their corresponding type.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "custom": { - "align": "auto", - "cellOptions": { - "barAlignment": 0, - "drawStyle": "line", - "gradientMode": "none", - "hideValue": false, - "lineInterpolation": "linear", - "lineStyle": { - "dash": [ - 10, - 10 - ], - "fill": "solid" - }, - "showPoints": "never", - "spanNulls": false, - "type": "sparkline" - }, - "inspect": false - }, - "displayName": "Block Type", - "fieldMinMax": false, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "block_type" - }, - "properties": [ - { - "id": "custom.width", - "value": 153 - }, - { - "id": "custom.cellOptions", - "value": { - "type": "auto" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Trend #Block Types" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 12, - "x": 12, - "y": 6 - }, - "id": 7, - "options": { - "cellHeight": "lg", - "footer": { - "countRows": false, - "enablePagination": false, - "fields": [ - "Value" - ], - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": false, - "sortBy": [ - { - "desc": true, - "displayName": "block_type" - } - ] - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m] offset -1m))/(60000/$__interval_ms)", - "format": "time_series", - "instant": false, - "legendFormat": "__auto", - "range": true, - "refId": "Block Types" - } - ], - "title": "Blocked Requests by Type", - "transformations": [ - { - "id": "timeSeriesTable", - "options": { - "A": { - "timeField": "Time" - }, - "Block Types": { - "stat": "sum", - "timeField": "Time" - } - } - } - ], - "type": "table" - }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 16 - }, - "id": 1, - "title": "Latency", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Percentiles of the application downstream latency over one minute.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "ms" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "25th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "50th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "95th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 0, - "y": 17 - }, - "id": 8, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "25th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "50th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "95th Percentile" - } - ], - "title": "Application Downstream Latency", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Percentiles of the Airlock Microgateway processing time over one minute.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "ms" - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "25th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "50th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "95th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } - } - ] - } - ] - }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 17 - }, - "id": 9, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "0.25 Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "0.5 Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "0.95 Percentile" - } - ], - "title": "Airlock Microgateway Processing Time", - "type": "timeseries" - } - ], - "refresh": "", - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "hide": 0, - "includeAll": true, - "label": "Operator Namespace", - "multi": true, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": ".*", - "skipUrlSync": false, - "sort": 0, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-24h", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway Overview", - "uid": "fdp5jb8fnrmyoa", - "version": 1, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/gke-values.yaml b/charts/airlock/microgateway/4.3.1/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway/4.3.1/openshift-values.yaml b/charts/airlock/microgateway/4.3.1/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway/4.3.1/questions.yml b/charts/airlock/microgateway/4.3.1/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway/4.3.1/templates/NOTES.txt b/charts/airlock/microgateway/4.3.1/templates/NOTES.txt index e38e3caa0..e8aa45888 100644 --- a/charts/airlock/microgateway/4.3.1/templates/NOTES.txt +++ b/charts/airlock/microgateway/4.3.1/templates/NOTES.txt @@ -1,34 +1,3 @@ -Thank you for installing Airlock Microgateway. -If you have not already done so, make sure that Airlock Microgateway CNI is also installed on the cluster. +Thank you for installing Airlock Microgateway CNI. -For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}. -Detailed CRD API reference documentation is also available at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}/api/crds. -{{ if .Values.crds.skipVersionCheck }} -- CRD version check skipped -{{- else }} -{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} -{{- if $outdatedCRDs -}} - {{- fail (printf ` - -Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. -Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: - -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts - -If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` - .Chart.AppVersion) - -}} -{{- end -}} -{{- end -}} -{{- if .Values.tests.enabled }} - {{- if .Values.operator.watchNamespaces -}} - {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) }} - {{- fail (printf ` - -To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. -` - .Release.Namespace) - -}} - {{- end -}} - {{- end -}} -{{- end }} \ No newline at end of file +For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" .}}. diff --git a/charts/airlock/microgateway/4.3.1/templates/_helpers.tpl b/charts/airlock/microgateway/4.3.1/templates/_helpers.tpl index 733ba9648..996491a87 100644 --- a/charts/airlock/microgateway/4.3.1/templates/_helpers.tpl +++ b/charts/airlock/microgateway/4.3.1/templates/_helpers.tpl @@ -1,16 +1,14 @@ {{/* Expand the name of the chart. -We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest explicit suffix is 14 characters. */}} -{{- define "airlock-microgateway.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Convert an image configuration object into an image ref string. */}} -{{- define "airlock-microgateway.image" -}} +{{- define "airlock-microgateway-cni.image" -}} {{- if .digest -}} {{- printf "%s@%s" .repository .digest -}} {{- else if .tag -}} @@ -22,19 +20,19 @@ Convert an image configuration object into an image ref string. {{/* Create a default fully qualified app name. -We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest implicit suffix is 27 characters. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. If release name contains chart name it will be used as a full name. */}} -{{- define "airlock-microgateway.fullname" -}} +{{- define "airlock-microgateway-cni.fullname" -}} {{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} {{- else }} {{- $name := default .Chart.Name .Values.nameOverride }} {{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} {{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} {{- end }} {{- end }} {{- end }} @@ -42,112 +40,62 @@ If release name contains chart name it will be used as a full name. {{/* Create chart name and version as used by the chart label. */}} -{{- define "airlock-microgateway.chart" -}} +{{- define "airlock-microgateway-cni.chart" -}} {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Common labels */}} -{{- define "airlock-microgateway.sharedLabels" -}} -helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} {{- if .Chart.AppVersion }} app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} {{- end }} app.kubernetes.io/managed-by: {{ .Release.Service }} -app.kubernetes.io/part-of: {{ .Chart.Name }} {{- with .Values.commonLabels }} {{ toYaml .}} {{- end }} {{- end }} {{/* -Common Selector labels +Common labels without component */}} -{{- define "airlock-microgateway.sharedSelectorLabels" -}} -app.kubernetes.io/instance: {{ .Release.Name }} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} {{- end }} {{/* -Restricted Container Security Context +Selector labels */}} -{{- define "airlock-microgateway.restrictedSecurityContext" -}} -allowPrivilegeEscalation: false -privileged: false -runAsNonRoot: true -capabilities: - drop: ["ALL"] -readOnlyRootFilesystem: true -seccompProfile: - type: RuntimeDefault +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} {{- end }} -{{/* Precondition: May only be used if AppVersion is isSemver */}} -{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} -{{- $version := (semver .Chart.AppVersion) -}} -{{- if $version.Prerelease -}} ->= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} -{{- else -}} ->= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 -{{- end -}} -{{- end -}} +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} -{{- define "airlock-microgateway.outdatedCRDs" -}} -{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} - {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} - {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} - {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} - {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} - {{- $isOutdated := false -}} - {{- if $crd -}} - {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} - {{- $isOutdated = true -}} - {{- if hasKey $crd.metadata "labels" -}} - {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} - {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} - {{- if (semverCompare $supportedVersion $crdVersion) }} - {{- $isOutdated = false -}} - {{- end }} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if $isOutdated }} -{{ base $path }} - {{- end }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{- define "airlock-microgateway.isSemver" -}} +{{- define "airlock-microgateway-cni.isSemver" -}} {{- regexMatch `^(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-]+)*))?$` . -}} {{- end -}} -{{- define "airlock-microgateway.docsVersion" -}} -{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} {{- $version := (semver .Chart.AppVersion) -}} {{- $version.Major }}.{{ $version.Minor -}} {{- else -}} {{- print "latest" -}} {{- end -}} {{- end -}} - -{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} -{{- $list := list -}} -{{- with .matchLabels -}} - {{- range $key, $value := . -}} - {{- $list = append $list (printf "%s=%s" $key $value) -}} - {{- end -}} -{{- end -}} -{{- with .matchExpressions -}} - {{- range . -}} - {{- if has .operator (list "In" "NotIn") -}} - {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} - {{- else if eq .operator "Exists" -}} - {{- $list = append $list .key -}} - {{- else if eq .operator "DoesNotExist" -}} - {{- $list = append $list (printf "!%s" .key) -}} - {{- end -}} - {{- end -}} -{{- end -}} -{{- join "," $list -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/clusterrole.yaml b/charts/airlock/microgateway/4.3.1/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/clusterrolebinding.yaml b/charts/airlock/microgateway/4.3.1/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/configmap.yaml b/charts/airlock/microgateway/4.3.1/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway/4.3.1/templates/daemonset.yaml b/charts/airlock/microgateway/4.3.1/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway/4.3.1/templates/network-attachment-definition.yaml b/charts/airlock/microgateway/4.3.1/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.3.1/templates/operator/_operator_helpers.tpl deleted file mode 100644 index a540ff9f4..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/_operator_helpers.tpl +++ /dev/null @@ -1,42 +0,0 @@ -{{/* -Create a default fully qualified name for operator components. -*/}} -{{- define "airlock-microgateway.operator.fullname" -}} -{{ include "airlock-microgateway.fullname" . }}-operator -{{- end }} - - -{{/* -Common operator labels -*/}} -{{- define "airlock-microgateway.operator.labels" -}} -{{ include "airlock-microgateway.sharedLabels" . }} -{{ include "airlock-microgateway.operator.selectorLabels" . }} -{{- end }} - -{{/* -Operator Selector labels -*/}} -{{- define "airlock-microgateway.operator.selectorLabels" -}} -{{ include "airlock-microgateway.sharedSelectorLabels" . }} -app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-operator -app.kubernetes.io/component: controller -{{- end }} - -{{/* -Create the name of the service account to use for the operator -*/}} -{{- define "airlock-microgateway.operator.serviceAccountName" -}} -{{- if .Values.operator.serviceAccount.create }} -{{- default (include "airlock-microgateway.operator.fullname" .) .Values.operator.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.operator.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -ServiceMonitor metrics regex pattern for leader only metrics -*/}} -{{- define "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" -}} -^(microgateway_license|microgateway_sidecars).*$ -{{- end }} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.3.1/templates/operator/_rbac.gen.tpl deleted file mode 100644 index 83b314cbc..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/_rbac.gen.tpl +++ /dev/null @@ -1,237 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator rbac permission rules -*/}} -{{- define "airlock-microgateway-operator.rbacRules" -}} -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - pods - verbs: - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - "" - resources: - - pods/finalizers - verbs: - - update -- apiGroups: - - "" - resources: - - pods/status - verbs: - - patch - - update -- apiGroups: - - "" - resources: - - secrets - verbs: - - create - - delete - - get - - list - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - accesscontrols - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - contentsecurities - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - denyrules - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyclusters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - envoyhttpfilters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - graphqls - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - headerrewrites - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - identitypropagations - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - limits - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcrelyingparties - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - openapis - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - parsers - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - redisproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sessionhandlings - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways - verbs: - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/finalizers - verbs: - - update -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - telemetries - verbs: - - get - - list - - watch -{{- end }} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/dashboard-configmap.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/dashboard-configmap.yaml deleted file mode 100644 index b71ac89b6..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/dashboard-configmap.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if .Values.dashboards.create -}} -{{- range $instance := (keys .Values.dashboards.instances | sortAlpha) -}} -{{- $dashboard := get $.Values.dashboards.instances $instance -}} -{{- if $dashboard.create }} ---- -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.fullname" $ }}-dashboard-{{ $instance | lower }} - namespace: {{ $.Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.dashboards.config.grafana.dashboardLabel -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - annotations: - {{- with $.Values.dashboards.config.grafana.folderAnnotation -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - {{- with $.Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -data: - {{- printf "%s.json" $instance | nindent 2 }}: |- - {{- ($.Files.Get (printf "dashboards/%s.json" $instance)) | nindent 4 -}} -{{- end -}} -{{- end -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/deployment.yaml deleted file mode 100644 index db340cdec..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/deployment.yaml +++ /dev/null @@ -1,143 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.operator.replicaCount }} - {{- with .Values.operator.updateStrategy }} - strategy: - {{- toYaml . | trim | nindent 4 }} - {{- end }} - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - checksum/config: {{ include (print $.Template.BasePath "/operator/configmap.yaml") . | sha256sum }} - kubectl.kubernetes.io/default-container: manager - {{- with mustMerge .Values.operator.podAnnotations .Values.commonAnnotations}} - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 8 }} - {{- with .Values.operator.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - containers: - - args: - - --config=operator_config.yaml - env: - - name: ENGINE_IMAGE - value: {{ include "airlock-microgateway.image" .Values.engine.image }} - - name: NETWORK_VALIDATOR_IMAGE - value: {{ include "airlock-microgateway.image" .Values.networkValidator.image }} - - name: SESSION_AGENT_IMAGE - value: {{ include "airlock-microgateway.image" .Values.sessionAgent.image }} - - name: OPERATOR_NAME - valueFrom: - fieldRef: - fieldPath: metadata.name - - name: OPERATOR_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - image: {{ include "airlock-microgateway.image" .Values.operator.image }} - imagePullPolicy: {{ .Values.operator.image.pullPolicy }} - livenessProbe: - httpGet: - path: /healthz - port: 8081 - initialDelaySeconds: 15 - periodSeconds: 20 - timeoutSeconds: 5 - name: manager - ports: - - containerPort: 9443 - name: webhook-server - protocol: TCP - - containerPort: 13377 - name: xds-server - protocol: TCP - - containerPort: 8080 - protocol: TCP - - containerPort: 8081 - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: 8081 - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - {{- with .Values.operator.resources }} - resources: - {{- toYaml . | nindent 10 }} - {{- end }} - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 10 }} - terminationMessagePath: /dev/termination-log - terminationMessagePolicy: FallbackToLogsOnError - volumeMounts: - - mountPath: /tmp/k8s-webhook-server/serving-certs - name: cert - readOnly: true - - mountPath: /opt/airlock/license/ - name: airlock-microgateway-license - readOnly: true - - mountPath: /operator_config.yaml - name: operator-config - subPath: operator_config.yaml - - mountPath: /sidecar/engine_container_template.yaml - name: operator-config - subPath: engine_container_template.yaml - - mountPath: /sidecar/network_validator_container_template.yaml - name: operator-config - subPath: network_validator_container_template.yaml - - mountPath: /sidecar/session_agent_container_template.yaml - name: operator-config - subPath: session_agent_container_template.yaml - - mountPath: /engine_bootstrap_config_template.yaml - name: operator-config - subPath: engine_bootstrap_config_template.yaml - securityContext: - runAsNonRoot: true - serviceAccountName: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - terminationGracePeriodSeconds: 10 - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - name: cert - secret: - defaultMode: 420 - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert - - name: airlock-microgateway-license - secret: - defaultMode: 292 - optional: true - secretName: {{ .Values.license.secretName }} - - configMap: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - name: operator-config diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/manager-role.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/manager-role.yaml deleted file mode 100644 index 90335bcfe..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/manager-role.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" . -}} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" $ }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/manager-rolebinding.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/manager-rolebinding.yaml deleted file mode 100644 index ae99cfb7b..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/manager-rolebinding.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" $ }} - namespace: {{ $.Release.Namespace }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/metrics-service.yaml deleted file mode 100644 index 34d23f6d6..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/metrics-service.yaml +++ /dev/null @@ -1,47 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} ---- -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-leader-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - operator.microgateway.airlock.com/isLeader: "true" - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/mutating-webhook.yaml deleted file mode 100644 index 311f9726a..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/mutating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.mutatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/podmonitor.yaml deleted file mode 100644 index 1fe34fcb3..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/podmonitor.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.engine.sidecar.podMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ include "airlock-microgateway.fullname" . }}-engine - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.engine.sidecar.podMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - any: true - selector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" - microgateway.airlock.com/managedBy: {{ .Release.Namespace }} - podMetricsEndpoints: - - targetPort: 19002 - path: /metrics - scheme: http -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/role.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/role.yaml deleted file mode 100644 index 5378be8ef..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/role.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/rolebinding.yaml deleted file mode 100644 index bafec1015..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/rolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/selfsigned-issuer.yaml deleted file mode 100644 index 466c56338..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/selfsigned-issuer.yaml +++ /dev/null @@ -1,13 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selfSigned: {} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/serviceaccount.yaml deleted file mode 100644 index 434d7e9d3..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/serviceaccount.yaml +++ /dev/null @@ -1,13 +0,0 @@ -{{- if .Values.operator.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with mustMerge .Values.operator.serviceAccount.annotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/servicemonitor.yaml deleted file mode 100644 index ff85a9a31..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/servicemonitor.yaml +++ /dev/null @@ -1,60 +0,0 @@ -{{- if .Values.operator.serviceMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - matchExpressions: - - { key: "operator.microgateway.airlock.com/isLeader", operator: DoesNotExist } - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: drop ---- -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - operator.microgateway.airlock.com/isLeader: "true" - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: keep -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/serving-certificate.yaml deleted file mode 100644 index 60b92e1e2..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/serving-certificate.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - dnsNames: - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc.cluster.local - issuerRef: - kind: Issuer - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/validating-webhook.yaml deleted file mode 100644 index 5d6b4396b..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/validating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.validatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/webhook-service.yaml deleted file mode 100644 index 477ea839f..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/webhook-service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-webhook - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: https - name: webhook - port: 443 - protocol: TCP - targetPort: 9443 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.3.1/templates/operator/xds-service.yaml deleted file mode 100644 index 81b41acf5..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/operator/xds-service.yaml +++ /dev/null @@ -1,24 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-xds - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: grpc - name: xds - port: 13377 - protocol: TCP - targetPort: 13377 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" diff --git a/charts/airlock/microgateway/4.3.1/templates/scc-role.yaml b/charts/airlock/microgateway/4.3.1/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/scc-rolebinding.yaml b/charts/airlock/microgateway/4.3.1/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/serviceaccount.yaml b/charts/airlock/microgateway/4.3.1/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway/4.3.1/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.3.1/templates/tests/rbac.yaml index 93bd4cd1b..744799333 100644 --- a/charts/airlock/microgateway/4.3.1/templates/tests/rbac.yaml +++ b/charts/airlock/microgateway/4.3.1/templates/tests/rbac.yaml @@ -2,142 +2,63 @@ apiVersion: v1 kind: ServiceAccount metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests --- apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests roleRef: apiGroup: rbac.authorization.k8s.io kind: Role - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" subjects: - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" --- apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests rules: - apiGroups: - - microgateway.airlock.com + - "apps" resources: - - sidecargateways + - daemonsets resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + - {{ include "airlock-microgateway-cni.fullname" . }} verbs: - - get - - list - - watch - - delete + - get + - watch + - list - apiGroups: - - microgateway.airlock.com + - "" resources: - - sidecargateways + - pods + - pods/log verbs: - - create + - get + - list +{{- if .Values.rbac.createSCCRole }} - apiGroups: - - "" - resources: - - events - verbs: - - list -- apiGroups: - - "apps" - resources: - - deployments + - security.openshift.io resourceNames: - - "{{ include "airlock-microgateway.operator.fullname" . }}" - verbs: - - get - - list - - watch -- apiGroups: - - "apps" + - privileged resources: - - statefulsets - - statefulsets/scale - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + - securitycontextconstraints verbs: - - get - - list - - watch - - patch -- apiGroups: - - "" - resources: - - pods - - pods/log - - pods/status - - pods/attach - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" - - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" - - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" - verbs: - - get - - list - - create - - watch - - delete -- apiGroups: - - "" - resources: - - pods - verbs: - - create -{{- if .Values.operator.watchNamespaceSelector }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -subjects: - - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -rules: -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list -{{- end }} + - use +{{- end -}} {{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/templates/tests/service.yaml b/charts/airlock/microgateway/4.3.1/templates/tests/service.yaml deleted file mode 100644 index 30ddc278d..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/tests/service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: v1 -kind: Service -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-service" - namespace: {{ .Release.Namespace }} - labels: - app: test-service - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - selector: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} - ports: - - name: http - port: 8080 - targetPort: 8080 -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.3.1/templates/tests/statefulset.yaml deleted file mode 100644 index 710a7b9f6..000000000 --- a/charts/airlock/microgateway/4.3.1/templates/tests/statefulset.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: apps/v1 -kind: StatefulSet -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - serviceName: nginx - replicas: 0 - selector: - matchLabels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - k8s.v1.cni.cncf.io/networks: default/airlock-microgateway-cni - labels: - sidecar.microgateway.airlock.com/inject: "true" - sidecar.istio.io/inject: "false" - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 8 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 8 }} - spec: - containers: - - image: cgr.dev/chainguard/nginx - name: nginx - ports: - - containerPort: 8080 - volumeMounts: - - mountPath: /var/lib/nginx/tmp/ - name: nginx-tmp - - mountPath: /var/run - name: nginx-run - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 12 }} - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - emptyDir: {} - name: nginx-tmp - - emptyDir: {} - name: nginx-run -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.3.1/templates/tests/test-install.yaml index ab82abea7..12d8c8de7 100644 --- a/charts/airlock/microgateway/4.3.1/templates/tests/test-install.yaml +++ b/charts/airlock/microgateway/4.3.1/templates/tests/test-install.yaml @@ -2,14 +2,11 @@ apiVersion: v1 kind: Pod metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" namespace: {{ .Release.Namespace }} labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - sidecar.istio.io/inject: "false" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} annotations: helm.sh/hook: test helm.sh/hook-delete-policy: before-hook-creation @@ -19,209 +16,88 @@ spec: - name: test image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true command: - sh - -c - | set -eu - clean_up() { - echo "" - echo "### Clean up test resources" - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - echo "" - echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s - sleep 3s - echo "" - } - fail() { + echo "Error: ${1}" echo "" - echo "### Error: ${1}" - echo "" - - if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then - echo "" - echo 'Microgateway Sidecargateway status:' - kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true - echo "" - echo "" - fi - - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then - echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" - kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true - echo "" - echo "" - echo 'Logs of Nginx container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true - echo "" - echo "" - # Wait for engine logs - sleep 10s - echo 'Logs of Microgateway Engine container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true - fi - + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer exit 1 } - create_sidecargateway() { - # create SidecarGateway resource for testing purposes - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done - kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - } - - {{- if .Values.operator.watchNamespaceSelector }} - echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" - if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then - labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') - fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` - .Release.Namespace - (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) - }} + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' fi - echo "" - {{- end }} - trap clean_up EXIT - echo "" - - echo "### Waiting for Microgateway Operator Deployments to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ - deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then - fail 'Timout occurred' + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' fi - echo "" - echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" - # scale to zero replicas to ensure no pods are present from previous runs - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s - echo "" - - echo "### Waiting for backend pod" - i=0 - while true; do - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then - break - elif [ $i -gt 3 ]; then - fail 'Pod not ready' - fi - sleep 2s - i=$((i+1)) - done - - echo "### Checking Microgateway Engine sidecar container was injected" - if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then - fail 'Microgateway Engine sidecar container not injected' + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' fi - echo "True" - echo "" - echo "### Checking for valid license" - i=0 - while true; do - if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then - break - elif [ $i -gt 30 ]; then - fail 'Microgateway license is missing or invalid' - fi - sleep 2s - i=$((i+1)) - done - echo "True" - echo "" + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac - echo "### Create SidecarGateway resource for testing" - if ! create_sidecargateway ; then - fail 'Creation of SidecarGateway resource failed' - fi - echo "" - - echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then - fail 'Timout occurred' - fi - echo "" - - echo "### Waiting for 'engine-config-valid' condition" - if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then - fail 'Configuration was never accepted by the Microgateway Engine' - fi - sleep 5s - echo "" - echo "" - - echo "### Checking whether a valid request is successful and returns HTTP status code '200'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "200 OK"; then - fail 'A valid request was not successful' - fi - echo "" - echo "" - - echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "400 Bad Request"; then - fail 'A malicious request was not blocked' - fi - echo "" - echo "" - - echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" - exit 0 - serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir {{- end -}} diff --git a/charts/airlock/microgateway/4.3.1/values.schema.json b/charts/airlock/microgateway/4.3.1/values.schema.json index 173d6b084..e087bd700 100644 --- a/charts/airlock/microgateway/4.3.1/values.schema.json +++ b/charts/airlock/microgateway/4.3.1/values.schema.json @@ -14,15 +14,6 @@ "commonAnnotations": { "$ref": "#/definitions/StringMap" }, - "crds": { - "type": "object", - "properties": { - "skipVersionCheck": { - "type": "boolean" - } - }, - "additionalProperties": false - }, "imagePullSecrets": { "type": "array", "items": { @@ -39,304 +30,120 @@ "additionalProperties": true } }, - "operator": { + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { "type": "object", "properties": { - "replicaCount": { - "type": "integer", - "minimum": 0 - }, - "updateStrategy": { - "$ref": "#/definitions/UpdateStrategy" - }, - "image": { - "$ref": "#/definitions/Image" - }, - "podAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "podLabels": { - "$ref": "#/definitions/StringMap" - }, - "serviceAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "serviceLabels": { - "$ref": "#/definitions/StringMap" - }, - "resources": { - "type": "object" - }, - "nodeSelector": { - "$ref": "#/definitions/StringMap" - }, - "tolerations": { - "type": "array", - "items": { - "type": "object" - } - }, - "affinity": { - "type": "object" - }, - "config": { - "type": "object", - "properties": { - "logLevel": { - "type": "string", - "enum": [ - "debug", - "info", - "warn", - "error" - ] - } - }, - "required": [ - "logLevel" - ], - "additionalProperties": false - }, - "serviceAccount": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "annotations": { - "$ref": "#/definitions/StringMap" - }, - "name": { - "type": "string" - } - }, - "required": [ - "annotations", - "create", - "name" - ], - "additionalProperties": false - }, - "watchNamespaces": { - "type": "array", - "items": { - "type": "string" - } - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - }, - "rbac": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - }, - "serviceMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "oneOf": [ - { - "properties": { - "watchNamespaces": { - "minItems": 1 - }, - "watchNamespaceSelector": { - "additionalProperties": false - } - } - }, - { - "properties": { - "watchNamespaces": { - "maxItems": 0 - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - } - } - } - ], - "required": [ - "affinity", - "config", - "image", - "updateStrategy", - "nodeSelector", - "podAnnotations", - "podLabels", - "rbac", - "replicaCount", - "resources", - "serviceAccount", - "serviceAnnotations", - "serviceLabels", - "serviceMonitor", - "tolerations" - ], - "additionalProperties": false - }, - "engine": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - }, - "sidecar": { - "type": "object", - "properties":{ - "podMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "required": [ - "podMonitor" - ], - "additionalProperties": false - } - }, - "required": [ - "image", - "resources", - "sidecar" - ], - "additionalProperties": false - }, - "networkValidator": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - } - }, - "required": [ - "image" - ], - "additionalProperties": false - }, - "sessionAgent": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - } - }, - "required": [ - "image", - "resources" - ], - "additionalProperties": false - }, - "license": { - "type": "object", - "properties": { - "secretName": { - "type": "string", - "minLength": 1 - } - }, - "required": [ - "secretName" - ], - "additionalProperties": false - }, - "dashboards": { - "type": "object", - "properties" : { "create": { "type": "boolean" }, - "config": { - "type": "object", - "properties": { - "grafana": { - "type": "object", - "properties": { - "folderAnnotation": { - "$ref": "#/definitions/NameValuePair" - }, - "dashboardLabel": { - "$ref": "#/definitions/NameValuePair" - } - }, - "required": [ - "folderAnnotation", - "dashboardLabel" - ], - "additionalProperties": false - } - }, - "required": [ - "grafana" - ], - "additionalProperties": false - }, - "instances": { - "type": "object", - "properties": { - "overview": { - "$ref": "#/definitions/DashboardInstance" - }, - "license" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockMetrics" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockLogs" : { - "$ref": "#/definitions/DashboardInstance" - } - }, - "required": [ - "overview", - "license", - "blockMetrics", - "blockLogs" - ], - "additionalProperties": false + "createSCCRole": { + "type": "boolean" } }, "required": [ "create", - "config", - "instances" + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" ], "additionalProperties": false }, @@ -357,18 +164,22 @@ } }, "required": [ + "affinity", "commonAnnotations", "commonLabels", - "crds", - "engine", + "config", "fullnameOverride", + "image", "imagePullSecrets", - "license", + "multusNetworkAttachmentDefinition", "nameOverride", - "operator", - "networkValidator", - "sessionAgent", - "dashboards", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", "tests" ], "additionalProperties": false, @@ -409,132 +220,6 @@ "tag" ], "additionalProperties": false - }, - "LabelSelector": { - "type": "object", - "properties": { - "matchExpressions": { - "type": "array", - "items": { - "type": "object", - "required": [ - "key", - "operator" - ], - "properties": { - "key": { - "type": "string" - }, - "operator": { - "type": "string" - }, - "values": { - "type": "array", - "items": { - "type": "string" - } - } - }, - "additionalProperties": false - } - }, - "matchLabels": { - "$ref": "#/definitions/StringMap" - } - }, - "additionalProperties": false - }, - "UpdateStrategy": { - "type": "object", - "oneOf" : [ - { - "properties": { - "type": { - "$ref": "#/definitions/RecreateType" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - }, - { - "properties": { - "type": { - "$ref": "#/definitions/RollingUpdateType" - }, - "rollingUpdate": { - "$ref": "#/definitions/RollingUpdate" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - } - ] - }, - "RecreateType": { - "type": "string", - "enum": [ - "Recreate" - ] - }, - "RollingUpdateType": { - "type": "string", - "enum": [ - "RollingUpdate" - ] - }, - "RollingUpdate": { - "type": "object", - "properties": { - "maxSurge": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - }, - "maxUnavailable": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - } - }, - "anyOf": [ - {"required": ["maxSurge"]}, - {"required": ["maxUnavailable"]} - ], - "additionalProperties": false - }, - "DashboardInstance" : { - "type" : "object", - "properties" : { - "create" : { - "type" : "boolean" - } - }, - "required" : [ - "create" - ], - "additionalProperties": false - }, - "NameValuePair" : { - "type" : "object", - "properties" : { - "name" : { - "type": "string", - "minLength": 1 - }, - "value" : { - "type" : "string", - "minLength": 1 - } - }, - "required" : [ - "name", - "value" - ], - "additionalProperties": false } } } diff --git a/charts/airlock/microgateway/4.3.1/values.yaml b/charts/airlock/microgateway/4.3.1/values.yaml index 120df1946..8f518138a 100644 --- a/charts/airlock/microgateway/4.3.1/values.yaml +++ b/charts/airlock/microgateway/4.3.1/values.yaml @@ -1,4 +1,4 @@ -# -- Allows overriding the name to use instead of "microgateway". +# -- Allows overriding the name to use instead of "microgateway-cni". nameOverride: "" # -- Allows overriding the name to use as full name of resources. fullnameOverride: "" @@ -10,203 +10,75 @@ commonAnnotations: {} imagePullSecrets: [] # - name: myRegistryKeySecretName -crds: - # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. - # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster - # when performing a "helm install/upgrade". - skipVersionCheck: false -operator: - # -- Number of replicas for the operator Deployment. - replicaCount: 2 - # -- Specifies the operator update strategy. - updateStrategy: - type: RollingUpdate - # Specifies the Airlock Microgateway Operator image. - image: - # -- Image repository from which to pull the Airlock Microgateway Operator image. - repository: "quay.io/airlock/microgateway-operator" - # -- Image tag to pull. - tag: "4.3.1" - # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). - # Overrides tag when specified. - digest: "sha256:84b6eb914103d4c62024d9f761b7dd4371ea3ba8996fb04095d87ebfaf3db2bb" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Annotations to add to all Pods. - podAnnotations: {} - # -- Labels to add to all Pods. - podLabels: {} - # -- Annotations to add to the Service. - serviceAnnotations: {} - # prometheus.io/scrape: "true" - # prometheus.io/port: "8080" - - # -- Labels to add to the Service. - serviceLabels: {} - # -- Resource restrictions to apply to the operator container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 1000m - # memory: 512Mi - # requests: - # cpu: 100m - # memory: 512Mi - - # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. - nodeSelector: {} - # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. - tolerations: [] - # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. - affinity: {} - # Parameters for the operator configuration. - config: - # -- Operator application log level. - logLevel: "info" - # Configures the generation of the ServiceAccount. - serviceAccount: - # -- Whether a ServiceAccount should be created. - create: true - # -- Annotations to add to the ServiceAccount. - annotations: {} - # -- Name of the ServiceAccount to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - # -- Allows to restrict the operator to specific namespaces, depending on your needs. - # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). - # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. - # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. - # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. - # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. - # Please note that this feature requires a Premium license. - watchNamespaces: [] - # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. - # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. - # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). - # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. - # Please note that this feature requires a Premium license. - watchNamespaceSelector: {} - # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. - # matchLabels: - # microgateway.airlock.com/enable: "true" - # matchExpressions: - # - { key: environment, operator: NotIn, values: [dev] } - - # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. - rbac: - # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. - create: true - # Configures the generation of a Prometheus Operator ServiceMonitor. - serviceMonitor: - # -- Whether to create a ServiceMonitor resource for monitoring. - create: false - # -- Labels to add to the ServiceMonitor. - labels: {} - # release: "" -engine: - # Specifies the Airlock Microgateway Engine image. - image: - # -- Image repository from which to pull the Airlock Microgateway Engine image. - repository: "quay.io/airlock/microgateway-engine" - # -- Image tag to pull. - tag: "4.3.1" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:6be782cc3f3206bfa61f462812d2a495e114ae13c59a7cdaed7ca406d5bc1b01" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Engine container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 500m - # memory: 128Mi - # requests: - # cpu: 10m - # memory: 40Mi - - # Additional configuration when deployed as a sidecar. - sidecar: - # Configures the generation of a Prometheus Operator PodMonitor. - podMonitor: - # -- Whether to create a PodMonitor resource for monitoring. - create: false - # -- Labels to add to the PodMonitor. - labels: {} - # release: "" -networkValidator: - # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. - image: - # -- Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. - repository: "cgr.dev/chainguard/netcat" - # -- Image tag to pull. - tag: "" - # -- SHA256 image digest to pull (in the format "sha256:6626ab44066867687baa7bfcabedafce5adc50446be1207c90c3b211bd922f84"). - # Overrides tag when specified. - digest: "sha256:6626ab44066867687baa7bfcabedafce5adc50446be1207c90c3b211bd922f84" - # -- Pull policy for this image. - pullPolicy: IfNotPresent -sessionAgent: - # Specifies the Airlock Microgateway Session Agent image. - image: - # -- Image repository from which to pull the Airlock Microgateway Session Agent image. - repository: "quay.io/airlock/microgateway-session-agent" - # -- Image tag to pull. - tag: "4.3.1" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:d62bdb16c74d340a81791be1696d620950d8232437676910bb6e5548411f2afd" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 150m - # memory: 32Mi - # requests: - # cpu: 10m - # memory: 8Mi -license: - # -- Name of the secret containing the "microgateway-license.txt" key. - secretName: "airlock-microgateway-license" -# Creates dashboards in the form of ConfigMaps that can be imported -# by Grafana using its sidecar setup. -dashboards: - # -- Whether to create any ConfigMaps containing Grafana dashboards to import. +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.3.1" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:bdd216c8a8c56a0eee0134f67772cbd75358640a0685cf5d71add653abb2c53b" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. create: false - config: - # Configures the necessary label and annotations along with their values - # to enable Grafana to correctly identify the ConfigMaps containing - # dashboards and file them within a dedicated folder in the dashboard overview. - # These settings need to match the Grafana sidecar configuration. - grafana: - folderAnnotation: - # -- Name of the annotation containing the folder name to file dashboards into. - name: "grafana_folder" - # -- Name of the folder dashboards are filed into within the Grafana UI. - value: "Airlock Microgateway" - dashboardLabel: - # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. - name: "grafana_dashboard" - # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. - value: "1" - instances: - # Available dashboard instances that can be individually created/deployed. - overview: - # -- Whether to create the overview dashboard. - create: true - license: - # -- Whether to create the license dashboard. - create: true - blockMetrics: - # -- Whether to create the block metrics dashboard. - create: true - blockLogs: - # -- Whether to create the block logs dashboard. - create: true -# Check whether the installation of the Airlock Microgateway Helm Chart was successful. -# Requires a secret with a valid Airlock Microgateway license key already to be present. + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system tests: # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). # If set to false, `helm test` will not run any tests. diff --git a/charts/airlock/microgateway/4.3.3/.helmignore b/charts/airlock/microgateway/4.3.3/.helmignore index 101ff5ac5..8561d2892 100644 --- a/charts/airlock/microgateway/4.3.3/.helmignore +++ b/charts/airlock/microgateway/4.3.3/.helmignore @@ -21,8 +21,7 @@ .idea/ *.tmproj .vscode/ -# CRDs kustomization.yaml -/crds/kustomization.yaml + # Helm unit tests /tests /validation diff --git a/charts/airlock/microgateway/4.3.3/Chart.yaml b/charts/airlock/microgateway/4.3.3/Chart.yaml index c168f9d77..f22c19bb6 100644 --- a/charts/airlock/microgateway/4.3.3/Chart.yaml +++ b/charts/airlock/microgateway/4.3.3/Chart.yaml @@ -9,15 +9,15 @@ annotations: - name: Airlock Microgateway Forum url: https://forum.airlock.com/ catalog.cattle.io/certified: partner - catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/display-name: Airlock Microgateway CNI catalog.cattle.io/kube-version: '>=1.25.0-0' - catalog.cattle.io/release-name: microgateway - charts.openshift.io/name: Airlock Microgateway + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI apiVersion: v2 appVersion: 4.3.3 -description: A Helm chart for deploying the Airlock Microgateway +description: A Helm chart for deploying the Airlock Microgateway CNI plugin home: https://www.airlock.com/en/microgateway -icon: file://assets/icons/microgateway.svg +icon: file://assets/icons/microgateway-cni.svg keywords: - WAF - Web Application Firewall @@ -30,14 +30,13 @@ keywords: - Filtering - DevSecOps - shift left -- control plane -- Operator +- CNI kubeVersion: '>=1.25.0-0' maintainers: - email: support@airlock.com name: Airlock url: https://www.airlock.com/ -name: microgateway +name: microgateway-cni sources: - https://github.com/airlock/microgateway type: application diff --git a/charts/airlock/microgateway/4.3.3/README.md b/charts/airlock/microgateway/4.3.3/README.md index c98085da1..685c4f1f8 100644 --- a/charts/airlock/microgateway/4.3.3/README.md +++ b/charts/airlock/microgateway/4.3.3/README.md @@ -1,4 +1,4 @@ -# Airlock Microgateway +# Airlock Microgateway CNI ![Version: 4.3.3](https://img.shields.io/badge/Version-4.3.3-informational?style=flat-square) ![AppVersion: 4.3.3](https://img.shields.io/badge/AppVersion-4.3.3-informational?style=flat-square) @@ -40,58 +40,43 @@ Check the official documentation at **[docs.airlock.com](https://docs.airlock.co The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. ## Prerequisites -* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) -* [Airlock Microgateway License](#obtain-airlock-microgateway-license) -* [cert-manager](https://cert-manager.io/) * [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) -In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. -For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. -### Obtain Airlock Microgateway License -1. Either request a community or premium license - * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) - * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) -2. Check your inbox and save the license file microgateway-license.txt locally. - -> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. -### Deploy cert-manager -```bash -helm repo add jetstack https://charts.jetstack.io -helm install cert-manager jetstack/cert-manager --version '1.15.1' -n cert-manager --create-namespace --set crds.enabled=true --wait -``` - -## Deploy Airlock Microgateway Operator - -> This guide assumes a microgateway-license.txt file is present in the working directory. - -1. Install CRDs and Operator. +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). ```bash - # Create namespace - kubectl create namespace airlock-microgateway-system - - # Install License - kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt - - # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) - helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.3.3' --wait + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.3/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.3/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). 2. (Recommended) You can verify the correctness of the installation with `helm test`. ```bash - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.3' - helm test airlock-microgateway -n airlock-microgateway-system --logs - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.3' + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.3' ``` -### Upgrading CRDs - -The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. -CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: -```bash -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.3.3 --server-side --force-conflicts -``` - -**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. ## Support @@ -104,61 +89,33 @@ For the community edition, check our **[Airlock community forum](https://forum.a | Key | Type | Default | Description | |-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | | commonAnnotations | object | `{}` | Annotations to add to all resources. | | commonLabels | object | `{}` | Labels to add to all resources. | -| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | -| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | -| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | -| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | -| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | -| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | -| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | -| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | -| engine.image.digest | string | `"sha256:3c0ebee0b560c8699723bfa433cd601b04b190c384e031d3789b83287fab7a9b"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | -| engine.image.tag | string | `"4.3.3"` | Image tag to pull. | -| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | -| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | -| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | | fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:16317b9a8430059c15175673ad53e31d9e882a1d1af6576214eb1534d8ea6937"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.3.3"` | Image tag to pull. | | imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | -| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | -| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | -| networkValidator.image.digest | string | `"sha256:6051975a14c51b9d3b525a06004d62a4d323c08ca58e3468343095a55a42fff2"` | SHA256 image digest to pull (in the format "sha256:6051975a14c51b9d3b525a06004d62a4d323c08ca58e3468343095a55a42fff2"). Overrides tag when specified. | -| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| networkValidator.image.repository | string | `"cgr.dev/chainguard/netcat"` | Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. | -| networkValidator.image.tag | string | `""` | Image tag to pull. | -| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | -| operator.config.logLevel | string | `"info"` | Operator application log level. | -| operator.image.digest | string | `"sha256:6d3ebca355de0a67f0bf5f088a15b9410564e500033d3e1f534a2f49a05bf4c3"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | -| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | -| operator.image.tag | string | `"4.3.3"` | Image tag to pull. | -| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | -| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | -| operator.podLabels | object | `{}` | Labels to add to all Pods. | -| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | -| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | -| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | -| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | -| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | -| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | -| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | -| operator.serviceLabels | object | `{}` | Labels to add to the Service. | -| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | -| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | -| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | -| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | -| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | -| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | -| sessionAgent.image.digest | string | `"sha256:994bf4117adb74da4e05c22ffc168d9844bc68efa6a7fb96d73e849d1ef67b56"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | -| sessionAgent.image.tag | string | `"4.3.3"` | Image tag to pull. | -| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | | tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | ## License diff --git a/charts/airlock/microgateway/4.3.3/app-readme.md b/charts/airlock/microgateway/4.3.3/app-readme.md deleted file mode 100644 index e32cac025..000000000 --- a/charts/airlock/microgateway/4.3.3/app-readme.md +++ /dev/null @@ -1,28 +0,0 @@ -# Airlock Microgateway - -*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* - -## Features -* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. -* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction -* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication -* Content security filters for protecting against known attacks (OWASP Top 10) -* Access control to allow only authenticated users to access the protected services -* API security features like JSON parsing or OpenAPI specification enforcement - -For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. - -## Requirements -* [Airlock Microgateway CNI Helm Chart](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Also available as Rancher Chart) -* [Airlock Microgateway License](https://github.com/airlock/microgateway?tab=readme-ov-file#obtain-airlock-microgateway-license) (After obtaining the license install it according to the [documentation](https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-operator)) -* [cert-manager](https://cert-manager.io/docs/installation/) - -## Documentation and links - -Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. - -* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) -* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) -* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) -* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) -* [GitHub](https://github.com/airlock/microgateway) \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/accesscontrols.microgateway.airlock.com.yaml deleted file mode 100644 index b6f1ab384..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/accesscontrols.microgateway.airlock.com.yaml +++ /dev/null @@ -1,124 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: accesscontrols.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: AccessControl - listKind: AccessControlList - plural: accesscontrols - singular: accesscontrol - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControl specifies the options to perform access control with a Microgateway Engine container. - 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: Specifies how the Airlock Microgateway Engine performs access control. - properties: - policies: - description: Policies configures access control policies. - items: - properties: - authorization: - description: Authorization configures how requests are authorized. An empty object value {} disables authorization. - properties: - authentication: - description: Authentication specifies that clients need to be authenticated with the provided method. - properties: - oidc: - description: OIDC configures client authentication using OpenID Connect. - properties: - oidcRelyingPartyRef: - description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - oidcRelyingPartyRef - type: object - type: object - type: object - identityPropagation: - description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. - properties: - actions: - description: Actions specifies the propagation actions. - items: - properties: - identityPropagationRef: - description: IdentityPropagationRef selects an IdentityPropagation to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - identityPropagationRef - type: object - type: array - onFailure: - description: |- - OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: - _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. - enum: - - Pass - type: string - required: - - actions - - onFailure - type: object - required: - - authorization - type: object - maxItems: 1 - minItems: 1 - type: array - required: - - policies - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/contentsecurities.microgateway.airlock.com.yaml deleted file mode 100644 index 05e059f8a..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/contentsecurities.microgateway.airlock.com.yaml +++ /dev/null @@ -1,139 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: contentsecurities.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: ContentSecurity - listKind: ContentSecurityList - plural: contentsecurities - singular: contentsecurity - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: ContentSecurity specifies the options to secure an upstream web application with a Microgateway Engine container. - 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: Specifies the options to secure an upstream web application with a Microgateway Engine container. - properties: - apiProtection: - description: |- - APIProtection defines the relevant configurations to protect APIs. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - graphQLRef: - description: |- - GraphQLRef selects the relevant GraphQL configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - openAPIRef: - description: |- - OpenAPIRef selects the relevant OpenAPI configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - filter: - description: |- - Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests - to protect against various attack patterns. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - denyRulesRef: - description: |- - DenyRulesRef selects the relevant DenyRules configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - headerRewritesRef: - description: |- - HeaderRewritesRef selects the relevant HeaderRewrites. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - limitsRef: - description: |- - LimitsRef selects the relevant Limits configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - parserRef: - description: |- - ParserRef selects the relevant Parser configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/denyrules.microgateway.airlock.com.yaml deleted file mode 100644 index fddaa375d..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/denyrules.microgateway.airlock.com.yaml +++ /dev/null @@ -1,1804 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: denyrules.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: DenyRules - listKind: DenyRulesList - plural: denyrules - singular: denyrules - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - DenyRules configures request filtering using Airlock built-in and custom deny rules. - Deny rules establish a negative security model. They define prohibited patterns which, when a match is found in a request, lead to it being blocked from reaching the upstream web application. - To handle possible false positives, lower the security level or define fine-granular deny rule exceptions - If undefined, default settings are applied, designed to work with most upstream web application services. - 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: Specification of the desired deny rules behavior. - properties: - request: - description: Request configures deny rules for downstream requests. - properties: - builtIn: - description: BuiltIn configures the built-in deny rules. - properties: - exceptions: - description: Exceptions allows to define exceptions for specific requests and deny rules. - items: - description: |- - DenyRulesException defines an exception for deny rules. Exceptions may be defined by any or a combination of the following elements: blockedData (the request data causing a block) or requestConditions (properties of a request without taking into consideration the reason why a request has been blocked). - At least one of blockedData and requestConditions must be set. - properties: - blockedData: - description: BlockedData defines an exception based on the request data causing the block. - properties: - graphQL: - description: |- - GraphQL defines an exception based on a blocked GraphQL query. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header defines an exception based on a blocked header. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON defines an exception based on a blocked JSON property. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - jsonPath: - description: |- - JSONPath defines the JSONPath pattern to match the path within the JSON. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - key: - description: |- - Key defines the key of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter defines an exception based on a blocked parameter. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source defines the source of the parameter. - enum: - - Query - - Post - - Any - type: string - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path defines an exception based on the blocked path. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment defines an exception based on a blocked path segment. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - segments: - description: Segments defines the position of a segment within the path. - properties: - index: - description: Index specifies an exact path segment position by index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value defines the value of a path segment. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - type: object - requestConditions: - description: RequestConditions defines an exception based on a property of a request without taking into consideration the reason why a request has been blocked. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKeys: - description: RuleKeys restricts the exception to a set of deny rules. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - type: object - type: array - overrides: - description: Overrides allows to override the builtIn settings for specific deny rules. - items: - description: DenyRulesOverride allows to override the builtIn settings for specific deny rules. - properties: - conditions: - description: Conditions select which built-in deny rules' settings will be adjusted. - properties: - ruleKeys: - description: RuleKeys is a list of built-in deny rule names. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - types: - description: Types defines the type of attributes the override should be applied on. If Types are defined without any RuleKeys the override is applied to all deny rules. - items: - description: |- - A deny rule override type name can be any of the following values: - Header | - Parameter | - Path | - JSON | - GraphQL - enum: - - Header - - Parameter - - Path - - PathSegment - - JSON - - GraphQL - type: string - minItems: 0 - type: array - type: object - settings: - description: Settings override the corresponding properties for the selected rules. - properties: - level: - description: Level specifies the filter strength. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: array - settings: - description: Settings contains the keys which will be adjusted. - properties: - level: - default: Standard - description: Level represents a set of deny rules with different filter strengths. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - type: object - type: object - custom: - description: Custom allows configuring additional deny rules. - properties: - rules: - description: Rules defines list of additional deny rules. - items: - properties: - blockData: - description: BlockData specifies the request data which should cause a block. - properties: - graphQL: - description: |- - GraphQL specifies to block requests containing a matching GraphQL property. - At least one of field, argument and value must be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header specifies to block requests containing a matching header. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON specifies to block requests containing a matching JSON property in the body. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - key: - description: Key defines the key of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter specifies to block requests containing a matching parameter. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path specifies to block requests with a matching path. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - matcher: - description: Matcher specifies which path to block. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment specifies to block requests containing a matching path segment. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - segments: - description: |- - Segments restricts which path segments are filtered by this rule. - If not specified, all segments of a path are filtered. - properties: - index: - description: Index restricts the rule to the path segment at this index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value specifies which path segment values to block. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - value - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKey: - description: RuleKey defines a technical key for the deny rule. Must be unique. - minLength: 1 - pattern: ^[A-Z][A-Z0-9_]*$ - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - required: - - blockData - - ruleKey - type: object - type: array - x-kubernetes-list-map-keys: - - ruleKey - x-kubernetes-list-type: map - type: object - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/envoyclusters.microgateway.airlock.com.yaml deleted file mode 100644 index bb564f942..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/envoyclusters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: envoyclusters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyCluster - listKind: EnvoyClusterList - plural: envoyclusters - singular: envoycluster - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyCluster is an additional Envoy Cluster resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy cluster. - properties: - value: - description: Value defines the Envoy Cluster which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/envoyconfigurations.microgateway.airlock.com.yaml deleted file mode 100644 index b6147ae08..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/envoyconfigurations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,185 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: envoyconfigurations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyConfiguration - listKind: EnvoyConfigurationList - plural: envoyconfigurations - singular: envoyconfiguration - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - EnvoyConfiguration is the Schema for the envoyconfigurations API - {{% notice warning %}} EnvoyConfiguration resources may contain sensitive information and thus RBAC permissions should be granted with care. {{% /notice %}} - 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: EnvoyConfigurationSpec defines the desired state of EnvoyConfiguration - properties: - envoyResources: - properties: - clusters: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - endpoints: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - extensions: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - listeners: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - routes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - runtimes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - scopedRoutes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - secrets: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - envoyResourcesRaw: - description: |- - EnvoyResourcesRaw defines the desired state for each resource type. The resources are stored as zstd compressed JSON bytes. - For debugging purposes, the resources can be inspected with the following command: `kubectl get envoyconfiguration -ojsonpath='{.spec.envoyResourcesRaw}' | base64 -d | zstd -d | jq` - format: byte - type: string - nodeID: - description: '**Deprecated:** This field is now ignored as NodeID is always derived from the resource name.' - type: string - type: object - status: - description: EnvoyConfigurationStatus defines the observed state of EnvoyConfiguration - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of EnvoyConfiguration condition. - type: string - required: - - status - - type - type: object - type: array - status: - type: string - xds: - properties: - resourceTypes: - additionalProperties: - description: XdsResourceTypeSyncStatus defines the sync status of xDS for a specific resource type - properties: - errorMessage: - description: ErrorMessage defines an optional message why the currently served resources of this resource type are rejected by the client. - type: string - resources: - additionalProperties: - description: XdsResourceStatus defines the status of xDS for a specific resource - properties: - version: - description: Version defines the version which is currently served for this resource. - type: string - required: - - version - type: object - description: Resources defines the resources which are currently served for this resource type. - type: object - status: - description: Status defines the current sync status of this resource type. - type: string - version: - description: Version defines the version which is currently served for this resource type. - type: string - required: - - resources - - status - - version - type: object - description: ResourceTypes defines the sync statuses for each resource type. - type: object - version: - description: Version defines the version of the underlying xDS snapshot. - type: integer - required: - - version - type: object - required: - - status - - xds - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml deleted file mode 100644 index c5eaad364..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: envoyhttpfilters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyHTTPFilter - listKind: EnvoyHTTPFilterList - plural: envoyhttpfilters - singular: envoyhttpfilter - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyHTTPFilter is an additional Envoy HTTP Filter resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy HTTP filter. - properties: - value: - description: Value defines the HTTP filter which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/headerrewrites.microgateway.airlock.com.yaml deleted file mode 100644 index a9f832a2b..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/headerrewrites.microgateway.airlock.com.yaml +++ /dev/null @@ -1,759 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: headerrewrites.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: HeaderRewrites - listKind: HeaderRewritesList - plural: headerrewrites - singular: headerrewrites - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. - properties: - request: - description: Request defines manipulations on upstream request headers. - properties: - add: - description: Add defines which request headers will be added before forwarding to the upstream. - properties: - custom: - description: |- - Custom allows configuring additional upstream request headers. - Add selected headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which request headers will be forwarded to the upstream. - This can either be allHeaders or matchingHeaders. - Default: matchingHeaders: {...} - properties: - allHeaders: - description: AllHeaders specifies that all request headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which request headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - standardHeaders: - default: true - description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which request headers will be removed before forwarding to the upstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - alternativeForwardedHeaders: - default: true - description: |- - AlternativeForwardedHeaders removes downstream request headers which could potentially - be abused to alter the upstream's view of the remote connection. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - response: - description: Response defines manipulations on upstream response headers. - properties: - add: - description: Add defines which response headers will be added before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - csp: - default: true - description: |- - CSP sets a content security policy which allows only same-origin requests except for images - if the 'Content-Security-Policy' header is not set by the upstream. - type: boolean - featurePolicy: - default: false - description: |- - FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features - if the 'Feature-Policy' header is not set by the upstream. - **Deprecated:** Use permissionsPolicy instead. - type: boolean - hsts: - default: true - description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. - type: boolean - hstsPreload: - default: false - description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. - type: boolean - permissionsPolicy: - default: true - description: |- - PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features - if the 'Permissions-Policy' header is not set by the upstream. - type: boolean - referrerPolicy: - default: true - description: |- - ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests - if the 'Referrer-Policy' header is not set by the upstream. - type: boolean - xContentTypeOptions: - default: true - description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. - type: boolean - xFrameOptions: - default: true - description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which response headers will be forwarded to the downstream. - This can either be allHeaders or matchingHeaders. - Default: allHeaders: {} - properties: - allHeaders: - description: AllHeaders specifies that all response headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which response headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response header. - properties: - standardHeaders: - default: false - description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which response headers will be removed before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - auth: - description: Auth defines the categories of headers concerning authentication. - properties: - basic: - default: false - description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. - type: boolean - negotiate: - default: true - description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. - type: boolean - ntlm: - default: true - description: |- - NTLM removes upstream response headers that advise clients to authenticate with NTLM. - By default, these headers are removed, because NTLM pass-through is not supported. - type: boolean - type: object - informationLeakage: - description: InformationLeakage defines the categories of headers concerning information leakage. - properties: - application: - default: true - description: Application removes upstream response headers that leak information about the deployed software. - type: boolean - server: - default: true - description: Server removes upstream response headers that leak information about the server. - type: boolean - type: object - permissiveCors: - default: true - description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured remove operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - settings: - description: Settings configures the HeaderRewrites filter. - properties: - operationalMode: - default: Production - description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. - enum: - - Production - - Integration - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/identitypropagations.microgateway.airlock.com.yaml deleted file mode 100644 index 4610fe8b8..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/identitypropagations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,108 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: identitypropagations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: IdentityPropagation - listKind: IdentityPropagationList - plural: identitypropagations - singular: identitypropagation - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: IdentityPropagation specifies the desired identity propagation. - 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: Specification of the desired identity propagation. - properties: - header: - description: Header configures identity propagation via a request header. - properties: - name: - description: Name of the header to set. - minLength: 1 - type: string - value: - description: Value to propagate to the application. - properties: - source: - description: Source from which to extract the value. - properties: - metadata: - description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. - properties: - key: - description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. - minLength: 1 - type: string - namespace: - description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. - minLength: 1 - type: string - required: - - key - - namespace - type: object - oidc: - description: OIDC specifies to extract a value from the result of an OpenID Connect flow. - properties: - idToken: - description: IDToken specifies to extract the value from the OpenID Connect ID Token. - properties: - claim: - description: Claim selects the JWT claim from which to extract the value. - minLength: 1 - type: string - required: - - claim - type: object - required: - - idToken - type: object - type: object - required: - - source - type: object - required: - - name - - value - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/limits.microgateway.airlock.com.yaml deleted file mode 100644 index 727b02496..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/limits.microgateway.airlock.com.yaml +++ /dev/null @@ -1,651 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: limits.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Limits - listKind: LimitsList - plural: limits - singular: limits - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Limits contains the configuration for limits. - 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: Specification of the desired limits behavior. - properties: - request: - description: Request defines the limits for requests. - properties: - limited: - description: Limited enables limits on request scope. - properties: - exceptions: - description: Exceptions defines limit exceptions. - items: - description: LimitsException defines an exception for limits. - properties: - length: - description: Length defines an exception for length limits based on the data element exceeding the limit. - properties: - graphQL: - description: GraphQL defines a field, argument or value length limit exception for a GraphQL query. - properties: - argument: - description: |- - Argument restricts the exception to GraphQL queries with a matching argument of a field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field restricts the exception to GraphQL queries with a matching field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value restricts the exception to GraphQL queries with a matching argument value. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: JSON defines a key and value length limit exception for a JSON property. - properties: - jsonPath: - description: |- - JSONPath restricts the exception to JSON properties with a matching JSONPath. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - required: - - jsonPath - type: object - parameter: - description: Parameter defines a name and value length limit exception for a parameter. - properties: - name: - description: Name restricts the exception to parameters with a matching name. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source restricts the exception to parameters of this kind. - enum: - - Query - - Post - - Any - type: string - required: - - name - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this exception to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - type: object - type: array - general: - description: General defines general request limits. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the request body. It specifies the number of bytes (0 = unlimited). This limit is effective for any request not processed by one of the content parsers (e.g. json) as configured in the Parser CRD. **Note** This limit does not apply to WebSocket or gRPC traffic. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - pathLength: - anyOf: - - type: integer - - type: string - default: 1Ki - description: PathLength defines the maximum path length for all requests (parsed and unparsed). - 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 - graphQL: - description: GraphQL defines the limits for GraphQL requests. - properties: - nestingDepth: - default: 10 - description: NestingDepth defines the maximum depth of nesting for GraphQL objects. - format: int64 - type: integer - querySize: - anyOf: - - type: integer - - type: string - default: 1Ki - description: QuerySize defines the maximum size for GraphQL queries. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: "256" - description: ValueLength defines the maximum length for GraphQL values. - 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 - json: - description: JSON defines the limits for JSON requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the JSON request body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - elementCount: - default: 10000 - description: ElementCount defines the maximum number of keys and array items in the whole JSON document (recursive). - format: int64 - type: integer - keyCount: - default: 250 - description: KeyCount defines the maximum number of keys of a single JSON object (non-recursive). - format: int64 - type: integer - keyLength: - anyOf: - - type: integer - - type: string - default: "128" - description: KeyLength defines the maximum length for JSON keys. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - nestingDepth: - default: 100 - description: NestingDepth defines the maximum depth of nesting for JSON objects and JSON arrays. - format: int64 - type: integer - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for JSON values. - 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 - multipart: - description: Multipart defines the limits for Multipart requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the Multipart request body. It specifies the number of bytes (0 = unlimited). - 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 - parameter: - description: Parameter defines the limits for request parameters. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the form data body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - count: - default: 128 - description: Count defines the maximum number of request parameters. - format: int64 - type: integer - nameLength: - anyOf: - - type: integer - - type: string - default: "128" - description: NameLength defines the maximum length for parameter names. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for parameter values. - 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 - unlimited: - description: Unlimited disables all limits on request scope. - type: object - type: object - settings: - description: Settings configures the limits filter. - properties: - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a limit hits. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/oidcproviders.microgateway.airlock.com.yaml deleted file mode 100644 index 74acbf4da..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/oidcproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,305 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: oidcproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCProvider - listKind: OIDCProviderList - plural: oidcproviders - singular: oidcprovider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCProvider specifies an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - 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: Specification of an OpenID Provider. - properties: - static: - description: Static configures an OpenID Provider by explicitly specifying all endpoints. - properties: - endpoints: - description: Endpoints specifies the OpenID Provider endpoints. - properties: - authorization: - description: Authorization specifies the endpoint to which the authorization request is sent. - properties: - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - token: - description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. - properties: - tls: - description: TLS defines TLS settings. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: |- - Custom explicitly specifies how the server certificate should be verified. - Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines constraints the presented certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - disabled: - description: |- - Disabled specifies to trust any certificate without verification. - THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. - type: object - type: object - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - required: - - authorization - - token - type: object - required: - - endpoints - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml deleted file mode 100644 index baa26ebcc..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/oidcrelyingparties.microgateway.airlock.com.yaml +++ /dev/null @@ -1,224 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: oidcrelyingparties.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCRelyingParty - listKind: OIDCRelyingPartyList - plural: oidcrelyingparties - singular: oidcrelyingparty - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} - 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: Specification of the OIDC Relying Party configuration. - properties: - clientID: - description: ClientID specifies the OIDCRelyingParty "client_id". - minLength: 1 - type: string - credentials: - description: Credentials used for client authentication on the back-channel with the authorization server. - properties: - clientSecret: - description: ClientSecret authenticates with the client password issued by the OpenID Provider (OP). - properties: - method: - default: BasicAuth - description: Method specifies in which format the client secret is sent with the authorization request. - enum: - - BasicAuth - - FormURLEncoded - type: string - secretRef: - description: SecretRef specifies the kubernetes secret containing the client password with key "client.secret". - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - clientSecret - type: object - oidcProviderRef: - description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - pathMapping: - description: PathMapping configures the action matching. - properties: - logoutPath: - description: LogoutPath specifies which request paths should initiate a logout. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - redirectPath: - description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - logoutPath - - redirectPath - type: object - redirectURI: - description: |- - RedirectURI configures the "redirect_uri" parameter included in the authorization request. - May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. - minLength: 1 - type: string - required: - - clientID - - credentials - - oidcProviderRef - - pathMapping - - redirectURI - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/openapis.microgateway.airlock.com.yaml deleted file mode 100644 index 1c0928710..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/openapis.microgateway.airlock.com.yaml +++ /dev/null @@ -1,167 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: openapis.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OpenAPI - listKind: OpenAPIList - plural: openapis - singular: openapi - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: OpenAPI contains the configuration for the OpenAPI specification. - 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: Specification of the desired OpenAPI specification. - properties: - response: - description: Response defines the validation behaviour for responses. - properties: - secured: - description: Secured enables response checking. - properties: - validation: - default: Lax - description: Validation defines the validation mode for responses. - enum: - - Lax - - Strict - type: string - type: object - unsecured: - description: Unsecured disables response checking. - type: object - type: object - settings: - description: Settings defines the settings to configure OpenAPI specification enforcement. - properties: - logging: - description: Logging specifies the access log behavior. - properties: - maxFailedSubvalidations: - default: 10 - description: MaxFailedSubvalidations defines the maximum number of failed subvalidations being logged. - format: int64 - type: integer - type: object - schema: - description: Schema configures the OpenAPI specification. - properties: - source: - description: Source specifies the OpenAPI specification to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'openapi.json'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - validation: - description: Validation specifies the patterns for the validation behavior. - properties: - authentication: - description: Authentication defines the settings for the authentication scheme. - properties: - oAuth2: - description: OAuth2 specifies the OAuth2 parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - oidc: - description: Oidc specifies the OIDC parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - type: object - type: object - required: - - schema - type: object - required: - - settings - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/crds/redisproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/redisproviders.microgateway.airlock.com.yaml deleted file mode 100644 index 8c662a2d0..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/redisproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,159 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: redisproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: RedisProvider - listKind: RedisProviderList - plural: redisproviders - singular: redisprovider - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: RedisProvider contains a client configuration for connecting to a Redis database. - 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: Specification of a Redis database client configuration. - properties: - auth: - description: Auth specifies the Redis credentials. - properties: - password: - description: Password specifies the Redis password. - properties: - secretRef: - description: SecretRef selects the secret containing the Redis password under the key 'redis.password'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - username: - default: default - description: Username specifies the Redis username to authenticate with. - minLength: 1 - pattern: ^[^\s]+$ - type: string - required: - - password - type: object - mode: - description: Mode configures the redis deployment mode. - properties: - standalone: - description: Standalone specifies the standalone Redis instance to connect to. - properties: - host: - description: Host specifies the IP or hostname. - minLength: 1 - pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ - type: string - port: - default: 6379 - description: Port specifies the port. - maximum: 65535 - minimum: 1 - type: integer - required: - - host - type: object - type: object - timeouts: - description: Timeouts specifies the timeouts when interacting with the Redis endpoint. - properties: - connect: - default: 5s - description: Connect specifies the timeout for establishing a connection. - type: string - maxDuration: - default: 2s - description: MaxDuration specifies the response timeout. - type: string - type: object - tls: - description: TLS defines TLS settings. If not specified, TLS is disabled i.e. unencrypted TCP is used when connecting to the Redis instance. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: Custom explicitly specifies how the server certificate should be verified. - properties: - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - required: - - certificates - type: object - required: - - trustedCA - type: object - disabled: - description: 'Disabled specifies to trust any certificate without verification. THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. Note: This setting currently also disables TLS SNI.' - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching the host and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Session Agent’s base image. - type: object - type: object - type: object - required: - - mode - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/crds/sessionhandlings.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/sessionhandlings.microgateway.airlock.com.yaml deleted file mode 100644 index 72747df77..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/sessionhandlings.microgateway.airlock.com.yaml +++ /dev/null @@ -1,77 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: sessionhandlings.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SessionHandling - listKind: SessionHandlingList - plural: sessionhandlings - singular: sessionhandling - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - SessionHandling contains the configuration for session handling. - - - {{% notice warning %}} The Session Handling feature (required for OIDC) is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as high-availability Redis configurations (e.g. Sentinel/Cluster) are not yet supported. - {{% /notice %}} - 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: Specification of the desired session handling behavior. - properties: - persistence: - description: Persistence configures where to store the session state. - properties: - redisProviderRef: - description: RedisProviderRef specifies to cache session information in the provided Redis instance. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - redisProviderRef - type: object - required: - - persistence - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/sidecargateways.microgateway.airlock.com.yaml deleted file mode 100644 index 6e1c04a48..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/sidecargateways.microgateway.airlock.com.yaml +++ /dev/null @@ -1,758 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: sidecargateways.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SidecarGateway - listKind: SidecarGatewayList - plural: sidecargateways - singular: sidecargateway - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: SidecarGateway contains the configuration how to configure the Airlock Microgateway Engine when used as Sidecar Container within the Pod of an application. - 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: Specification of the desired sidecar gateway behavior. - properties: - applications: - description: Applications defines applications which run on different ports. - items: - properties: - containerPort: - default: 8080 - description: |- - ContainerPort refers to the container port. - This must be a valid port number, 0 < x < 65536. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - downstream: - description: Downstream defines the downstream configuration for this application - properties: - protocol: - description: |- - Protocol defines the exposed HTTP protocol version. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies that the protocol should be inferred. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies that the client is assumed to speak HTTP/1.1. - type: object - http2: - description: HTTP2 specifies that the client is assumed to speak HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - remoteIP: - description: |- - RemoteIP defines how the remote IP of a client is propagated. - Default: xff: {...} - properties: - connectionIP: - description: ConnectionIP configures to use the source IP address of the direct downstream connection. - type: object - customHeader: - description: CustomHeader specifies to use a custom header for remote IP extraction. - properties: - headerName: - description: HeaderName specifies the name of the custom header containing the remote IP. - minLength: 1 - type: string - required: - default: true - description: Required specifies if the custom header is required. If true and not available the request will be rejected with 403. - type: boolean - required: - - headerName - type: object - xff: - description: XFF configures to use the standard 'X-Forwarded-For' header for IP extraction. - properties: - numTrustedHops: - default: 1 - description: NumTrustedHops specifies to extract the client's originating IP from the nth rightmost entry in the X-Forwarded-For header. With the default value of 1, the IP is extracted from the rightmost entry. - format: int32 - minimum: 1 - type: integer - type: object - type: object - requestNormalizations: - description: RequestNormalizations defines a set of normalization actions which are applied to the request before route matching. - properties: - mergeSlashes: - default: true - description: MergeSlashes ensures that adjacent slashes in the path are merged into one. - type: boolean - normalizePath: - default: true - description: NormalizePath ensures normalization according to RFC 3986 without case normalization. - type: boolean - type: object - restrictions: - description: Restrictions defines restrictions for downstream. - properties: - http: - description: HTTP defines limits for the HTTP protocol. - properties: - headersLength: - anyOf: - - type: integer - - type: string - default: 60Ki - description: HeadersLength defines maximum size of all request headers combined. Requests that exceed this limit will receive a 431 response. - 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 - timeouts: - description: Timeouts defines timeouts for downstream - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - default: 5m - description: |- - Idle defines the settings for the idle timeout when no data is sent or received. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - maxDuration: - default: 5m - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - requestHeaders: - default: 10s - description: |- - RequestHeaders defines the duration before all request headers must be received. - A value of 0 will completely disable the timeout. - Default: 10s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - clientCertificate: - description: |- - ClientCertificate defines the TLS settings for verification of client certificates. - At most one of ignored, optional and required can be set. - Default: ignored: {} - properties: - ignored: - description: Ignored disables verification of the client certificate. - type: object - optional: - description: |- - Optional enables verification of the client certificate if one is presented. - In this mode only trustedCA and crl settings can be configured since certificatePinning and allowedSANs require a client certificate. - properties: - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - required: - - trustedCA - type: object - required: - description: |- - Required contains settings for client certificate verification. A client must present a valid certificate. - At least one of trustedCA and certificatePinning must be set. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines the constraints a client certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - type: object - enable: - default: false - description: Enable defines if the downstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - secretRef: - description: SecretRef defines the reference to the TLS server certificate (secret of type kubernetes.io/tls). - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - xfcc: - description: |- - XFCC defines the handling of X-Forwarded-Client-Cert header. Meaning of the possible values: - _Sanitize_: Do not send the XFCC header to the next hop. This is the default value. - _ForwardOnly_: When the client connection is mTLS (Mutual TLS), forward the XFCC header in the request. - _AppendAndForward_: When the client connection is mTLS, append the client certificate information to the request’s XFCC header and forward it. - _SanitizeAndSet_: When the client connection is mTLS, reset the XFCC header with the client certificate information and send it to the next hop. - _AlwaysForwardOnly_: Always forward the XFCC header in the request, regardless of whether the client connection is mTLS. - Note: When forwarding the XFCC header in the request you might have to adjust the header length restrictions (See sidecargateway.spec.applications.downstream.restrictions.http) - enum: - - Sanitize - - ForwardOnly - - AppendAndForward - - SanitizeAndSet - - AlwaysForwardOnly - type: string - type: object - type: object - envoyHTTPFilterRefs: - description: EnvoyHTTPFilterRefs selects the relevant EnvoyHTTPFilters. - properties: - prepend: - description: Prepend selects the relevant EnvoyHTTPFilters which are added before those configured by the Airlock Microgateway. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - type: object - routes: - description: Routes defines the security configurations for different paths. The first matching route (from top to bottom) applies. - items: - description: |- - SidecarGatewayApplicationRoute defines the security configurations for different paths. - At most one of secured and unsecured can be set. - Default: secured: {...} - properties: - pathPrefix: - default: / - description: PathPrefix defines the path prefix used during route selection. - minLength: 1 - type: string - secured: - description: Secured enables WAF processing for this route. - properties: - accessControlRef: - description: |- - AccessControlRef selects the relevant AccessControl configuration resource. - If undefined, Airlock Microgateway does not perform any access control. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - contentSecurityRef: - description: |- - ContentSecurityRef selects the relevant ContentSecurity configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - unsecured: - description: |- - Unsecured disables all WAF functionality and therefore protection for this route. - WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. - type: object - type: object - type: array - x-kubernetes-list-map-keys: - - pathPrefix - x-kubernetes-list-type: map - telemetryRef: - description: |- - TelemetryRef selects the relevant Telemetry configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - upstream: - description: Upstream defines the upstream configuration for this application - properties: - protocol: - description: |- - Protocol defines HTTP protocol version used to communicate with the upstream. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies to negotiate the protocol with TLS ALPN (if TLS is enabled) or, as a fallback, use the same protocol that is used by the downstream connection. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies to use HTTP/1.1. - type: object - http2: - description: HTTP2 specifies to use HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - timeouts: - description: Timeouts defines the timeout settings. - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - description: |- - Timeout defines the settings for http timeouts. If this setting is not specified, the value of applications[].downstream.timeouts.http.idle is inherited. - A value of 0 will completely disable the timeout. - type: string - maxDuration: - default: 15s - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - Default: 15s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - enable: - default: false - description: Enable defines if the upstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - type: object - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - containerPort - x-kubernetes-list-type: map - envoyClusterRefs: - description: EnvoyClusterRefs selects the relevant EnvoyClusters. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - podSelector: - description: PodSelector defines to which Pods the configuration will be applied to. - properties: - matchLabels: - additionalProperties: - type: string - description: MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels. - type: object - type: object - sessionHandlingRef: - description: SessionHandlingRef selects the SessionHandling configuration to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - applications - type: object - status: - description: Most recently observed status of the SidecarGateway which is populated by the system. This data is read-only and may not be up to date. - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of SidecarGateway condition. - type: string - required: - - status - - type - type: object - type: array - pods: - items: - properties: - envoyConfig: - description: EnvoyConfig indicates the name of the EnvoyConfig CR for the Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - status: - type: string - unmanagedPods: - items: - properties: - managedBy: - description: ManagedBy indicates the Airlock Microgateway Operator instance which manages this Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - required: - - status - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.3/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.3/crds/telemetries.microgateway.airlock.com.yaml deleted file mode 100644 index 3262cb1f0..000000000 --- a/charts/airlock/microgateway/4.3.3/crds/telemetries.microgateway.airlock.com.yaml +++ /dev/null @@ -1,96 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 - name: telemetries.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Telemetry - listKind: TelemetryList - plural: telemetries - singular: telemetry - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Telemetry contains the configuration for telemetry (logging, metrics & tracing). - 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: Specification of the desired telemetry behavior. - properties: - correlation: - description: Correlation defines the correlation aspects of Telemetry. - properties: - idSource: - description: IDSource specifies how an external correlation ID should be obtained for a request. If not specified, no correlation ID will be logged. - properties: - header: - description: Header specifies to extract the correlation ID from a request header. If the header is absent from a request, no correlation ID will be logged. - properties: - name: - default: X-Correlation-Id - description: Name of the header (case-insensitive) from which to extract the correlation ID. - minLength: 1 - type: string - type: object - required: - - header - type: object - request: - description: Request defines the request related correlation settings of Telemetry. - properties: - allowDownstreamRequestID: - default: true - description: AllowDownstreamRequestID defines whether trace sampling will consider a provided x-request-id. - type: boolean - alterRequestID: - default: true - description: AlterRequestID defines whether to alter the UUID to reflect the trace sampling decision. If disabled no modification to the UUID will be performed, this may break tracing in the upstream. - type: boolean - type: object - type: object - logging: - description: Logging defines the logging aspects of Telemetry. - properties: - accessLog: - description: AccessLog defines the access log settings of Telemetry. - properties: - format: - description: Format defines the Access Log format of the sidecar. - properties: - json: - description: JSON defines the Access Log format as JSON. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.3/dashboards/blockMetrics.json b/charts/airlock/microgateway/4.3.3/dashboards/blockMetrics.json deleted file mode 100644 index ba383d22e..000000000 --- a/charts/airlock/microgateway/4.3.3/dashboards/blockMetrics.json +++ /dev/null @@ -1,758 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "panel", - "id": "barchart", - "name": "Bar chart", - "version": "" - }, - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "description": "Metrics on requests blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 6, - "title": "Airlock Microgateway Block Metrics", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 1 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [ - { - "options": { - "match": "nan", - "result": { - "index": 0, - "text": "n/a" - } - }, - "type": "special" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 4, - "y": 1 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", - "fullMetaSearch": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Blocked Requests (%)", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "% Blocked Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "left", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 20, - "x": 0, - "y": 5 - }, - "id": 3, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "timezone": [ - "" - ], - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "instant": false, - "legendFormat": "Requests per second", - "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "hide": false, - "instant": false, - "legendFormat": "% Blocks", - "range": true, - "refId": "Blocks" - } - ], - "title": "Requests vs. % Blocks", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Blocked requests by block type.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "super-light-orange", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisGridShow": true, - "axisLabel": "", - "axisPlacement": "auto", - "fillOpacity": 80, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineWidth": 0, - "scaleDistribution": { - "type": "linear" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "fieldMinMax": false, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 11, - "w": 10, - "x": 0, - "y": 15 - }, - "id": 4, - "options": { - "barRadius": 0, - "barWidth": 0.8, - "fullHighlight": false, - "groupWidth": 0.7, - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "orientation": "horizontal", - "showValue": "never", - "stacking": "none", - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "asc" - }, - "xField": "block_type", - "xTickLabelRotation": 0, - "xTickLabelSpacing": 0 - }, - "pluginVersion": "10.4.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "instant": true, - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Block Type", - "transformations": [ - { - "id": "reduce", - "options": { - "includeTimeField": false, - "labelsToFields": true, - "mode": "seriesToRows", - "reducers": [ - "sum" - ] - } - } - ], - "type": "barchart" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Blocked requests by attack type, which are subsets of the various block types.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "light-orange", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "fillOpacity": 80, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineWidth": 1, - "scaleDistribution": { - "type": "linear" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 11, - "w": 10, - "x": 10, - "y": 15 - }, - "id": 5, - "options": { - "barRadius": 0, - "barWidth": 0.8, - "fullHighlight": false, - "groupWidth": 0.7, - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "orientation": "horizontal", - "showValue": "never", - "stacking": "none", - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - }, - "xField": "attack_type", - "xTickLabelRotation": 0, - "xTickLabelSpacing": 0 - }, - "pluginVersion": "10.4.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum by (attack_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "instant": true, - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Attack Type", - "transformations": [ - { - "id": "reduce", - "options": { - "labelsToFields": true, - "reducers": [ - "sum" - ] - } - } - ], - "type": "barchart" - } - ], - "refresh": "", - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "Datasource Prometheus", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, - "hide": 2, - "includeAll": false, - "label": "DS_LOKI", - "multi": false, - "name": "DS_LOKI", - "options": [], - "query": "loki", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "hide": 0, - "includeAll": true, - "label": "Operator Namespace", - "multi": true, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": ".*", - "skipUrlSync": false, - "sort": 0, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "hide": 0, - "includeAll": true, - "label": "Block Type", - "multi": true, - "name": "blockType", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - } - ] - }, - "time": { - "from": "now-24h", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": { - "hidden": false - }, - "timezone": "browser", - "title": "Airlock Microgateway Block Metrics", - "uid": "ddnqoczu7qvb4cdd3dd", - "version": 3, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/dashboards/license.json b/charts/airlock/microgateway/4.3.3/dashboards/license.json deleted file mode 100644 index b9d5777e2..000000000 --- a/charts/airlock/microgateway/4.3.3/dashboards/license.json +++ /dev/null @@ -1,521 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 0 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Expiry date of the Airlock Microgateway license associated with the selected operator.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "time: L" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 3, - "y": 0 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_expiry_timestamp_seconds{namespace=~\"${operator_namespace.regex}\"})*1000", - "instant": true, - "legendFormat": "Expiry Date (MM/DD/YYYY)", - "range": false, - "refId": "A" - } - ], - "title": "License Expiry Date", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of licensed requests for applications protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 7, - "y": 0 - }, - "id": 6, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_license_max_rq_count_per_month{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Licensed Requests", - "range": false, - "refId": "A" - } - ], - "title": "Licensed Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Estimated number of requests protected by Airlock Microgateway over 30 days based on the last 7 days.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 5, - "x": 11, - "y": 0 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d]))/7*30", - "instant": true, - "legendFormat": "Estimated Requests", - "range": false, - "refId": "A" - } - ], - "title": "Requests over 30 days (estimated)", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of requests per week processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 12, - "w": 16, - "x": 0, - "y": 4 - }, - "id": 5, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(avg_over_time(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d])[2m:30s]))", - "instant": false, - "legendFormat": "# Requests per week", - "range": true, - "refId": "A" - } - ], - "title": "Processed Requests per week", - "type": "timeseries" - } - ], - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "description": "", - "hide": 0, - "includeAll": false, - "label": "Operator Namespace", - "multi": false, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-7d", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway License", - "uid": "cdpq79bzrr01se", - "version": 2, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/gke-values.yaml b/charts/airlock/microgateway/4.3.3/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway/4.3.3/openshift-values.yaml b/charts/airlock/microgateway/4.3.3/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway/4.3.3/questions.yml b/charts/airlock/microgateway/4.3.3/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway/4.3.3/templates/NOTES.txt b/charts/airlock/microgateway/4.3.3/templates/NOTES.txt index 6e5ce218a..bb94ff521 100644 --- a/charts/airlock/microgateway/4.3.3/templates/NOTES.txt +++ b/charts/airlock/microgateway/4.3.3/templates/NOTES.txt @@ -1,47 +1,15 @@ -Thank you for installing Airlock Microgateway. +Thank you for installing Airlock Microgateway CNI. -Please ensure the following prerequisites are fulfilled: -* Cert-Manager is installed. - https://cert-manager.io/docs/installation/helm/ -* Airlock Microgateway CNI is also installed on the cluster. - https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni -* A valid Airlock Microgateway license is deployed in the Kubernetes secret 'airlock-microgateway-license'. - * Get a free Community license: https://airlock.com/en/microgateway-community - * Order a Premium license: https://airlock.com/en/microgateway-premium +Please ensure that the helm values'.config.cniNetDir' and '.config.cniBinDir' are configured for your Kubernetes distribution. +For further information, consider our manual https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }}. +The chapter 'Setup > Installation' describes how to set those settings correctly. Further information: -* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }} -* CRD API reference documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }}/api/crds +* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }} * Airlock Microgateway Labs: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=helm -{{- if .Values.crds.skipVersionCheck }} -Warning: CRD version check skipped -{{- else -}} -{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} -{{- if $outdatedCRDs -}} - {{- fail (printf ` - -Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. -Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: - -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts - -If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` - .Chart.AppVersion) - -}} -{{- end -}} -{{- end -}} -{{- if .Values.tests.enabled -}} - {{- if .Values.operator.watchNamespaces -}} - {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) -}} - {{- fail (printf ` - -To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. -` - .Release.Namespace) - -}} - {{- end -}} - {{- end -}} -{{- end }} +Next steps: +* Install Airlock Microgateway (if not done already) + https://artifacthub.io/packages/helm/airlock-microgateway/microgateway Your release version is {{ .Chart.Version }}. \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/_helpers.tpl b/charts/airlock/microgateway/4.3.3/templates/_helpers.tpl index 733ba9648..996491a87 100644 --- a/charts/airlock/microgateway/4.3.3/templates/_helpers.tpl +++ b/charts/airlock/microgateway/4.3.3/templates/_helpers.tpl @@ -1,16 +1,14 @@ {{/* Expand the name of the chart. -We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest explicit suffix is 14 characters. */}} -{{- define "airlock-microgateway.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Convert an image configuration object into an image ref string. */}} -{{- define "airlock-microgateway.image" -}} +{{- define "airlock-microgateway-cni.image" -}} {{- if .digest -}} {{- printf "%s@%s" .repository .digest -}} {{- else if .tag -}} @@ -22,19 +20,19 @@ Convert an image configuration object into an image ref string. {{/* Create a default fully qualified app name. -We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest implicit suffix is 27 characters. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. If release name contains chart name it will be used as a full name. */}} -{{- define "airlock-microgateway.fullname" -}} +{{- define "airlock-microgateway-cni.fullname" -}} {{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} {{- else }} {{- $name := default .Chart.Name .Values.nameOverride }} {{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} {{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} {{- end }} {{- end }} {{- end }} @@ -42,112 +40,62 @@ If release name contains chart name it will be used as a full name. {{/* Create chart name and version as used by the chart label. */}} -{{- define "airlock-microgateway.chart" -}} +{{- define "airlock-microgateway-cni.chart" -}} {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Common labels */}} -{{- define "airlock-microgateway.sharedLabels" -}} -helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} {{- if .Chart.AppVersion }} app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} {{- end }} app.kubernetes.io/managed-by: {{ .Release.Service }} -app.kubernetes.io/part-of: {{ .Chart.Name }} {{- with .Values.commonLabels }} {{ toYaml .}} {{- end }} {{- end }} {{/* -Common Selector labels +Common labels without component */}} -{{- define "airlock-microgateway.sharedSelectorLabels" -}} -app.kubernetes.io/instance: {{ .Release.Name }} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} {{- end }} {{/* -Restricted Container Security Context +Selector labels */}} -{{- define "airlock-microgateway.restrictedSecurityContext" -}} -allowPrivilegeEscalation: false -privileged: false -runAsNonRoot: true -capabilities: - drop: ["ALL"] -readOnlyRootFilesystem: true -seccompProfile: - type: RuntimeDefault +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} {{- end }} -{{/* Precondition: May only be used if AppVersion is isSemver */}} -{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} -{{- $version := (semver .Chart.AppVersion) -}} -{{- if $version.Prerelease -}} ->= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} -{{- else -}} ->= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 -{{- end -}} -{{- end -}} +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} -{{- define "airlock-microgateway.outdatedCRDs" -}} -{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} - {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} - {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} - {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} - {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} - {{- $isOutdated := false -}} - {{- if $crd -}} - {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} - {{- $isOutdated = true -}} - {{- if hasKey $crd.metadata "labels" -}} - {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} - {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} - {{- if (semverCompare $supportedVersion $crdVersion) }} - {{- $isOutdated = false -}} - {{- end }} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if $isOutdated }} -{{ base $path }} - {{- end }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{- define "airlock-microgateway.isSemver" -}} +{{- define "airlock-microgateway-cni.isSemver" -}} {{- regexMatch `^(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-]+)*))?$` . -}} {{- end -}} -{{- define "airlock-microgateway.docsVersion" -}} -{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} {{- $version := (semver .Chart.AppVersion) -}} {{- $version.Major }}.{{ $version.Minor -}} {{- else -}} {{- print "latest" -}} {{- end -}} {{- end -}} - -{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} -{{- $list := list -}} -{{- with .matchLabels -}} - {{- range $key, $value := . -}} - {{- $list = append $list (printf "%s=%s" $key $value) -}} - {{- end -}} -{{- end -}} -{{- with .matchExpressions -}} - {{- range . -}} - {{- if has .operator (list "In" "NotIn") -}} - {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} - {{- else if eq .operator "Exists" -}} - {{- $list = append $list .key -}} - {{- else if eq .operator "DoesNotExist" -}} - {{- $list = append $list (printf "!%s" .key) -}} - {{- end -}} - {{- end -}} -{{- end -}} -{{- join "," $list -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/clusterrole.yaml b/charts/airlock/microgateway/4.3.3/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/clusterrolebinding.yaml b/charts/airlock/microgateway/4.3.3/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/configmap.yaml b/charts/airlock/microgateway/4.3.3/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway/4.3.3/templates/daemonset.yaml b/charts/airlock/microgateway/4.3.3/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway/4.3.3/templates/network-attachment-definition.yaml b/charts/airlock/microgateway/4.3.3/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.3.3/templates/operator/_operator_helpers.tpl deleted file mode 100644 index a540ff9f4..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/_operator_helpers.tpl +++ /dev/null @@ -1,42 +0,0 @@ -{{/* -Create a default fully qualified name for operator components. -*/}} -{{- define "airlock-microgateway.operator.fullname" -}} -{{ include "airlock-microgateway.fullname" . }}-operator -{{- end }} - - -{{/* -Common operator labels -*/}} -{{- define "airlock-microgateway.operator.labels" -}} -{{ include "airlock-microgateway.sharedLabels" . }} -{{ include "airlock-microgateway.operator.selectorLabels" . }} -{{- end }} - -{{/* -Operator Selector labels -*/}} -{{- define "airlock-microgateway.operator.selectorLabels" -}} -{{ include "airlock-microgateway.sharedSelectorLabels" . }} -app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-operator -app.kubernetes.io/component: controller -{{- end }} - -{{/* -Create the name of the service account to use for the operator -*/}} -{{- define "airlock-microgateway.operator.serviceAccountName" -}} -{{- if .Values.operator.serviceAccount.create }} -{{- default (include "airlock-microgateway.operator.fullname" .) .Values.operator.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.operator.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -ServiceMonitor metrics regex pattern for leader only metrics -*/}} -{{- define "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" -}} -^(microgateway_license|microgateway_sidecars).*$ -{{- end }} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.3.3/templates/operator/_rbac.gen.tpl deleted file mode 100644 index 83b314cbc..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/_rbac.gen.tpl +++ /dev/null @@ -1,237 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator rbac permission rules -*/}} -{{- define "airlock-microgateway-operator.rbacRules" -}} -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - pods - verbs: - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - "" - resources: - - pods/finalizers - verbs: - - update -- apiGroups: - - "" - resources: - - pods/status - verbs: - - patch - - update -- apiGroups: - - "" - resources: - - secrets - verbs: - - create - - delete - - get - - list - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - accesscontrols - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - contentsecurities - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - denyrules - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyclusters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - envoyhttpfilters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - graphqls - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - headerrewrites - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - identitypropagations - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - limits - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcrelyingparties - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - openapis - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - parsers - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - redisproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sessionhandlings - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways - verbs: - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/finalizers - verbs: - - update -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - telemetries - verbs: - - get - - list - - watch -{{- end }} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.3.3/templates/operator/_webhooks.gen.tpl deleted file mode 100644 index 02e304890..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/_webhooks.gen.tpl +++ /dev/null @@ -1,339 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator mutating webhooks -*/}} -{{- define "airlock-microgateway-operator.mutatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /mutate-v1-pod - failurePolicy: Fail - name: mutate-pod.microgateway.airlock.com - reinvocationPolicy: IfNeeded - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -{{- end }} - -{{/* -Operator validating webhooks -*/}} -{{- define "airlock-microgateway-operator.validatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-v1-pod - failurePolicy: Fail - name: validate-pod.microgateway.airlock.com - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - - UPDATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-accesscontrol - failurePolicy: Fail - name: validate-accesscontrol.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - accesscontrols - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-denyrules - failurePolicy: Fail - name: validate-denyrules.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - denyrules - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoycluster - failurePolicy: Fail - name: validate-envoycluster.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyclusters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoyhttpfilter - failurePolicy: Fail - name: validate-envoyhttpfilter.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyhttpfilters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-graphql - failurePolicy: Fail - name: validate-graphql.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - graphqls - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-headerrewrites - failurePolicy: Fail - name: validate-headerrewrites.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - headerrewrites - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-identitypropagation - failurePolicy: Fail - name: validate-identitypropagation.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - identitypropagations - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-limits - failurePolicy: Fail - name: validate-limits.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - limits - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcprovider - failurePolicy: Fail - name: validate-oidcprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcrelyingparty - failurePolicy: Fail - name: validate-oidcrelyingparty.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcrelyingparties - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-openapi - failurePolicy: Fail - name: validate-openapi.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - openapis - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-parser - failurePolicy: Fail - name: validate-parser.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - parsers - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-redisprovider - failurePolicy: Fail - name: validate-redisprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - redisproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-sidecargateway - failurePolicy: Fail - name: validate-sidecargateway.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - sidecargateways - sideEffects: None -{{- end }} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/configmap.yaml deleted file mode 100644 index 95e52d7df..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/configmap.yaml +++ /dev/null @@ -1,394 +0,0 @@ -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -data: - engine_bootstrap_config_template.yaml: | - # Base configuration, admin interface on port 19000 - admin: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - dynamic_resources: - cds_config: - initial_fetch_timeout: 10s - resource_api_version: V3 - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - lds_config: - resource_api_version: V3 - initial_fetch_timeout: 10s - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - static_resources: - listeners: - - name: probe - address: - socket_address: - address: 0.0.0.0 - port_value: 19001 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: probe - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: probe - virtual_hosts: - - name: probe - domains: - - '*' - routes: - - name: ready - match: - path: /ready - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - - name: metrics - address: - socket_address: - address: 0.0.0.0 - port_value: 19002 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: metrics - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: metrics - virtual_hosts: - - name: metrics - domains: - - '*' - routes: - - name: metrics - match: - path: /metrics - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - prefix_rewrite: '/stats/prometheus' - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - clusters: - - name: xds_cluster - connect_timeout: 1s - type: STRICT_DNS - load_assignment: - cluster_name: xds_cluster - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: airlock-microgateway-operator-xds.$(OPERATOR_NAMESPACE).svc.cluster.local - port_value: 13377 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - transport_socket: - name: envoy.transport_sockets.tls - typed_config: - '@type': type.googleapis.com/envoy.extensions.transport_sockets.tls.v3.UpstreamTlsContext - common_tls_context: - tls_params: - tls_minimum_protocol_version: TLSv1_3 - tls_maximum_protocol_version: TLSv1_3 - validation_context_sds_secret_config: - name: validation_context_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/validation_context_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - tls_certificate_sds_secret_configs: - - name: tls_certificate_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/tls_certificate_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - - name: airlock_microgateway_engine_admin - connect_timeout: 1s - type: STATIC - load_assignment: - cluster_name: airlock_microgateway_engine_admin - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - stats_config: - stats_tags: - - tag_name: "block_type" - regex: "\\.(block_type\\.([^.]+))" - - tag_name: "attack_type" - regex: "\\.(attack_type\\.([^.]+))" - - tag_name: "envoy_cluster_name" - regex: "\\.(cluster\\.([^.]+))" - - tag_name: "version" - regex: "\\.(version\\.([^.]+))" - use_all_default_tags: true - overload_manager: - resource_monitors: - - name: "envoy.resource_monitors.global_downstream_max_connections" - typed_config: - "@type": type.googleapis.com/envoy.extensions.resource_monitors.downstream_connections.v3.DownstreamConnectionsConfig - max_active_downstream_connections: 50000 - bootstrap_extensions: - - name: airlock.bootstrap.engine_build_info - typed_config: - '@type': type.googleapis.com/airlock.extensions.bootstrap.stats.v1alpha.Stats - application_log_config: - log_format: - text_format: '{"@timestamp":"%Y-%m-%dT%T.%e%z","log":{"logger":"%n","level":"%l","origin":{"file":{"name":"%g","line":%#},"function":"%!"}},"event":{"module":"envoy","dataset":"envoy.application"},"process":{"pid":%P,"thread":{"id":%t}},"ecs":{"version":"8.5"},"message":"%j"}' - engine_container_template.yaml: | - name: "$(ENGINE_NAME)" - image: "$(ENGINE_IMAGE)" - imagePullPolicy: {{ .Values.engine.image.pullPolicy }} - args: - - "--config-path" - - "/etc/envoy/bootstrap_config.yaml" - - "--base-id" - - "$(BASE_ID)" - - "--file-flush-interval-msec" - - '1000' - - "--drain-time-s" - - '60' - - "--service-node" - - "$(POD_NAME).$(POD_NAMESPACE)" - - "--service-cluster" - - "$(APP_NAME).$(POD_NAMESPACE)" - - "--log-path" - - "/dev/stdout" - - "--log-level" - - "$(LOG_LEVEL)" - volumeMounts: - - name: airlock-microgateway-bootstrap-secret-volume - mountPath: /etc/envoy - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - - name: POD_IP - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: status.podIP - ports: - - containerPort: 13378 - protocol: TCP - - containerPort: 19001 - protocol: TCP - - containerPort: 19002 - protocol: TCP - livenessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 1 - readinessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.engine.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - session_agent_container_template.yaml: | - name: "$(SESSION_AGENT_NAME)" - image: "$(SESSION_AGENT_IMAGE)" - imagePullPolicy: {{ .Values.sessionAgent.image.pullPolicy }} - args: - - "--port" - - "19004" - - "--config-path" - - "/etc/microgateway-session-agent/config.json" - volumeMounts: - - name: airlock-microgateway-session-agent-volume - mountPath: /etc/microgateway-session-agent - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - ports: - - containerPort: 19004 - livenessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 5 - readinessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 5 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.sessionAgent.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - network_validator_container_template.yaml: | - name: "$(NETWORK_VALIDATOR_NAME)" - image: "$(NETWORK_VALIDATOR_IMAGE)" - imagePullPolicy: {{ .Values.networkValidator.image.pullPolicy }} - command: ["/bin/sh", "-c"] - args: - - |- - echo 'pong' | nc -v -l 127.0.0.1 13378 & - for i in 1 2 3; do - sleep 1s - if r=$(echo 'ping' | nc -v -q 0 127.0.0.1 19003) && [ $r == pong ]; then - echo -n 'Traffic redirection to Airlock Microgateway Engine is working.' > /dev/termination-log - exit 0 - fi - done - echo -en 'Traffic redirection to Airlock Microgateway Engine is not working.\nRestart the pod after ensuring that hostNetwork is disabled and a compatible Airlock Microgateway CNI version is installed on the node.\nCertain environments may also require additional configuration (see docs.airlock.com for more information).' > /dev/termination-log - exit 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - operator_config.yaml: | - apiVersion: config.airlock.com/v1alpha1 - kind: OperatorConfig - health: - healthProbeBindAddress: :8081 - metrics: - bindAddress: 0.0.0.0:8080 - webhook: - port: 9443 - deployment: - sidecar: - engineContainerTemplate: "/sidecar/engine_container_template.yaml" - networkValidatorContainerTemplate: "/sidecar/network_validator_container_template.yaml" - sessionAgentContainerTemplate: "/sidecar/session_agent_container_template.yaml" - engine: - bootstrapConfigTemplate: "/engine_bootstrap_config_template.yaml" - log: - level: {{ .Values.operator.config.logLevel }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaces: - selector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaces: - list: - {{- toYaml . | nindent 8 }} - {{- end }} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/dashboard-configmap.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/dashboard-configmap.yaml deleted file mode 100644 index b71ac89b6..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/dashboard-configmap.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if .Values.dashboards.create -}} -{{- range $instance := (keys .Values.dashboards.instances | sortAlpha) -}} -{{- $dashboard := get $.Values.dashboards.instances $instance -}} -{{- if $dashboard.create }} ---- -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.fullname" $ }}-dashboard-{{ $instance | lower }} - namespace: {{ $.Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.dashboards.config.grafana.dashboardLabel -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - annotations: - {{- with $.Values.dashboards.config.grafana.folderAnnotation -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - {{- with $.Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -data: - {{- printf "%s.json" $instance | nindent 2 }}: |- - {{- ($.Files.Get (printf "dashboards/%s.json" $instance)) | nindent 4 -}} -{{- end -}} -{{- end -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/deployment.yaml deleted file mode 100644 index db340cdec..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/deployment.yaml +++ /dev/null @@ -1,143 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.operator.replicaCount }} - {{- with .Values.operator.updateStrategy }} - strategy: - {{- toYaml . | trim | nindent 4 }} - {{- end }} - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - checksum/config: {{ include (print $.Template.BasePath "/operator/configmap.yaml") . | sha256sum }} - kubectl.kubernetes.io/default-container: manager - {{- with mustMerge .Values.operator.podAnnotations .Values.commonAnnotations}} - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 8 }} - {{- with .Values.operator.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - containers: - - args: - - --config=operator_config.yaml - env: - - name: ENGINE_IMAGE - value: {{ include "airlock-microgateway.image" .Values.engine.image }} - - name: NETWORK_VALIDATOR_IMAGE - value: {{ include "airlock-microgateway.image" .Values.networkValidator.image }} - - name: SESSION_AGENT_IMAGE - value: {{ include "airlock-microgateway.image" .Values.sessionAgent.image }} - - name: OPERATOR_NAME - valueFrom: - fieldRef: - fieldPath: metadata.name - - name: OPERATOR_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - image: {{ include "airlock-microgateway.image" .Values.operator.image }} - imagePullPolicy: {{ .Values.operator.image.pullPolicy }} - livenessProbe: - httpGet: - path: /healthz - port: 8081 - initialDelaySeconds: 15 - periodSeconds: 20 - timeoutSeconds: 5 - name: manager - ports: - - containerPort: 9443 - name: webhook-server - protocol: TCP - - containerPort: 13377 - name: xds-server - protocol: TCP - - containerPort: 8080 - protocol: TCP - - containerPort: 8081 - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: 8081 - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - {{- with .Values.operator.resources }} - resources: - {{- toYaml . | nindent 10 }} - {{- end }} - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 10 }} - terminationMessagePath: /dev/termination-log - terminationMessagePolicy: FallbackToLogsOnError - volumeMounts: - - mountPath: /tmp/k8s-webhook-server/serving-certs - name: cert - readOnly: true - - mountPath: /opt/airlock/license/ - name: airlock-microgateway-license - readOnly: true - - mountPath: /operator_config.yaml - name: operator-config - subPath: operator_config.yaml - - mountPath: /sidecar/engine_container_template.yaml - name: operator-config - subPath: engine_container_template.yaml - - mountPath: /sidecar/network_validator_container_template.yaml - name: operator-config - subPath: network_validator_container_template.yaml - - mountPath: /sidecar/session_agent_container_template.yaml - name: operator-config - subPath: session_agent_container_template.yaml - - mountPath: /engine_bootstrap_config_template.yaml - name: operator-config - subPath: engine_bootstrap_config_template.yaml - securityContext: - runAsNonRoot: true - serviceAccountName: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - terminationGracePeriodSeconds: 10 - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - name: cert - secret: - defaultMode: 420 - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert - - name: airlock-microgateway-license - secret: - defaultMode: 292 - optional: true - secretName: {{ .Values.license.secretName }} - - configMap: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - name: operator-config diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/manager-role.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/manager-role.yaml deleted file mode 100644 index 90335bcfe..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/manager-role.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" . -}} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" $ }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/manager-rolebinding.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/manager-rolebinding.yaml deleted file mode 100644 index ae99cfb7b..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/manager-rolebinding.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" $ }} - namespace: {{ $.Release.Namespace }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/metrics-service.yaml deleted file mode 100644 index 34d23f6d6..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/metrics-service.yaml +++ /dev/null @@ -1,47 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} ---- -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-leader-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - operator.microgateway.airlock.com/isLeader: "true" - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/mutating-webhook.yaml deleted file mode 100644 index 311f9726a..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/mutating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.mutatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/podmonitor.yaml deleted file mode 100644 index 1fe34fcb3..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/podmonitor.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.engine.sidecar.podMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ include "airlock-microgateway.fullname" . }}-engine - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.engine.sidecar.podMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - any: true - selector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" - microgateway.airlock.com/managedBy: {{ .Release.Namespace }} - podMetricsEndpoints: - - targetPort: 19002 - path: /metrics - scheme: http -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/role.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/role.yaml deleted file mode 100644 index 5378be8ef..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/role.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/rolebinding.yaml deleted file mode 100644 index bafec1015..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/rolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/selfsigned-issuer.yaml deleted file mode 100644 index 466c56338..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/selfsigned-issuer.yaml +++ /dev/null @@ -1,13 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selfSigned: {} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/serviceaccount.yaml deleted file mode 100644 index 434d7e9d3..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/serviceaccount.yaml +++ /dev/null @@ -1,13 +0,0 @@ -{{- if .Values.operator.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with mustMerge .Values.operator.serviceAccount.annotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/servicemonitor.yaml deleted file mode 100644 index ff85a9a31..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/servicemonitor.yaml +++ /dev/null @@ -1,60 +0,0 @@ -{{- if .Values.operator.serviceMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - matchExpressions: - - { key: "operator.microgateway.airlock.com/isLeader", operator: DoesNotExist } - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: drop ---- -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - operator.microgateway.airlock.com/isLeader: "true" - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: keep -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/serving-certificate.yaml deleted file mode 100644 index 60b92e1e2..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/serving-certificate.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - dnsNames: - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc.cluster.local - issuerRef: - kind: Issuer - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/validating-webhook.yaml deleted file mode 100644 index 5d6b4396b..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/validating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.validatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/webhook-service.yaml deleted file mode 100644 index 477ea839f..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/webhook-service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-webhook - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: https - name: webhook - port: 443 - protocol: TCP - targetPort: 9443 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.3.3/templates/operator/xds-service.yaml deleted file mode 100644 index 81b41acf5..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/operator/xds-service.yaml +++ /dev/null @@ -1,24 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-xds - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: grpc - name: xds - port: 13377 - protocol: TCP - targetPort: 13377 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" diff --git a/charts/airlock/microgateway/4.3.3/templates/scc-role.yaml b/charts/airlock/microgateway/4.3.3/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/scc-rolebinding.yaml b/charts/airlock/microgateway/4.3.3/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/serviceaccount.yaml b/charts/airlock/microgateway/4.3.3/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway/4.3.3/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.3.3/templates/tests/rbac.yaml index 93bd4cd1b..744799333 100644 --- a/charts/airlock/microgateway/4.3.3/templates/tests/rbac.yaml +++ b/charts/airlock/microgateway/4.3.3/templates/tests/rbac.yaml @@ -2,142 +2,63 @@ apiVersion: v1 kind: ServiceAccount metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests --- apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests roleRef: apiGroup: rbac.authorization.k8s.io kind: Role - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" subjects: - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" --- apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests rules: - apiGroups: - - microgateway.airlock.com + - "apps" resources: - - sidecargateways + - daemonsets resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + - {{ include "airlock-microgateway-cni.fullname" . }} verbs: - - get - - list - - watch - - delete + - get + - watch + - list - apiGroups: - - microgateway.airlock.com + - "" resources: - - sidecargateways + - pods + - pods/log verbs: - - create + - get + - list +{{- if .Values.rbac.createSCCRole }} - apiGroups: - - "" - resources: - - events - verbs: - - list -- apiGroups: - - "apps" - resources: - - deployments + - security.openshift.io resourceNames: - - "{{ include "airlock-microgateway.operator.fullname" . }}" - verbs: - - get - - list - - watch -- apiGroups: - - "apps" + - privileged resources: - - statefulsets - - statefulsets/scale - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + - securitycontextconstraints verbs: - - get - - list - - watch - - patch -- apiGroups: - - "" - resources: - - pods - - pods/log - - pods/status - - pods/attach - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" - - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" - - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" - verbs: - - get - - list - - create - - watch - - delete -- apiGroups: - - "" - resources: - - pods - verbs: - - create -{{- if .Values.operator.watchNamespaceSelector }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -subjects: - - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -rules: -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list -{{- end }} + - use +{{- end -}} {{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/templates/tests/service.yaml b/charts/airlock/microgateway/4.3.3/templates/tests/service.yaml deleted file mode 100644 index 30ddc278d..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/tests/service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: v1 -kind: Service -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-service" - namespace: {{ .Release.Namespace }} - labels: - app: test-service - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - selector: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} - ports: - - name: http - port: 8080 - targetPort: 8080 -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.3.3/templates/tests/statefulset.yaml deleted file mode 100644 index 710a7b9f6..000000000 --- a/charts/airlock/microgateway/4.3.3/templates/tests/statefulset.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: apps/v1 -kind: StatefulSet -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - serviceName: nginx - replicas: 0 - selector: - matchLabels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - k8s.v1.cni.cncf.io/networks: default/airlock-microgateway-cni - labels: - sidecar.microgateway.airlock.com/inject: "true" - sidecar.istio.io/inject: "false" - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 8 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 8 }} - spec: - containers: - - image: cgr.dev/chainguard/nginx - name: nginx - ports: - - containerPort: 8080 - volumeMounts: - - mountPath: /var/lib/nginx/tmp/ - name: nginx-tmp - - mountPath: /var/run - name: nginx-run - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 12 }} - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - emptyDir: {} - name: nginx-tmp - - emptyDir: {} - name: nginx-run -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.3.3/templates/tests/test-install.yaml index ab82abea7..12d8c8de7 100644 --- a/charts/airlock/microgateway/4.3.3/templates/tests/test-install.yaml +++ b/charts/airlock/microgateway/4.3.3/templates/tests/test-install.yaml @@ -2,14 +2,11 @@ apiVersion: v1 kind: Pod metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" namespace: {{ .Release.Namespace }} labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - sidecar.istio.io/inject: "false" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} annotations: helm.sh/hook: test helm.sh/hook-delete-policy: before-hook-creation @@ -19,209 +16,88 @@ spec: - name: test image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true command: - sh - -c - | set -eu - clean_up() { - echo "" - echo "### Clean up test resources" - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - echo "" - echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s - sleep 3s - echo "" - } - fail() { + echo "Error: ${1}" echo "" - echo "### Error: ${1}" - echo "" - - if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then - echo "" - echo 'Microgateway Sidecargateway status:' - kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true - echo "" - echo "" - fi - - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then - echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" - kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true - echo "" - echo "" - echo 'Logs of Nginx container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true - echo "" - echo "" - # Wait for engine logs - sleep 10s - echo 'Logs of Microgateway Engine container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true - fi - + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer exit 1 } - create_sidecargateway() { - # create SidecarGateway resource for testing purposes - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done - kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - } - - {{- if .Values.operator.watchNamespaceSelector }} - echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" - if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then - labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') - fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` - .Release.Namespace - (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) - }} + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' fi - echo "" - {{- end }} - trap clean_up EXIT - echo "" - - echo "### Waiting for Microgateway Operator Deployments to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ - deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then - fail 'Timout occurred' + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' fi - echo "" - echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" - # scale to zero replicas to ensure no pods are present from previous runs - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s - echo "" - - echo "### Waiting for backend pod" - i=0 - while true; do - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then - break - elif [ $i -gt 3 ]; then - fail 'Pod not ready' - fi - sleep 2s - i=$((i+1)) - done - - echo "### Checking Microgateway Engine sidecar container was injected" - if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then - fail 'Microgateway Engine sidecar container not injected' + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' fi - echo "True" - echo "" - echo "### Checking for valid license" - i=0 - while true; do - if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then - break - elif [ $i -gt 30 ]; then - fail 'Microgateway license is missing or invalid' - fi - sleep 2s - i=$((i+1)) - done - echo "True" - echo "" + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac - echo "### Create SidecarGateway resource for testing" - if ! create_sidecargateway ; then - fail 'Creation of SidecarGateway resource failed' - fi - echo "" - - echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then - fail 'Timout occurred' - fi - echo "" - - echo "### Waiting for 'engine-config-valid' condition" - if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then - fail 'Configuration was never accepted by the Microgateway Engine' - fi - sleep 5s - echo "" - echo "" - - echo "### Checking whether a valid request is successful and returns HTTP status code '200'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "200 OK"; then - fail 'A valid request was not successful' - fi - echo "" - echo "" - - echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "400 Bad Request"; then - fail 'A malicious request was not blocked' - fi - echo "" - echo "" - - echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" - exit 0 - serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir {{- end -}} diff --git a/charts/airlock/microgateway/4.3.3/values.schema.json b/charts/airlock/microgateway/4.3.3/values.schema.json index 173d6b084..e087bd700 100644 --- a/charts/airlock/microgateway/4.3.3/values.schema.json +++ b/charts/airlock/microgateway/4.3.3/values.schema.json @@ -14,15 +14,6 @@ "commonAnnotations": { "$ref": "#/definitions/StringMap" }, - "crds": { - "type": "object", - "properties": { - "skipVersionCheck": { - "type": "boolean" - } - }, - "additionalProperties": false - }, "imagePullSecrets": { "type": "array", "items": { @@ -39,304 +30,120 @@ "additionalProperties": true } }, - "operator": { + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { "type": "object", "properties": { - "replicaCount": { - "type": "integer", - "minimum": 0 - }, - "updateStrategy": { - "$ref": "#/definitions/UpdateStrategy" - }, - "image": { - "$ref": "#/definitions/Image" - }, - "podAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "podLabels": { - "$ref": "#/definitions/StringMap" - }, - "serviceAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "serviceLabels": { - "$ref": "#/definitions/StringMap" - }, - "resources": { - "type": "object" - }, - "nodeSelector": { - "$ref": "#/definitions/StringMap" - }, - "tolerations": { - "type": "array", - "items": { - "type": "object" - } - }, - "affinity": { - "type": "object" - }, - "config": { - "type": "object", - "properties": { - "logLevel": { - "type": "string", - "enum": [ - "debug", - "info", - "warn", - "error" - ] - } - }, - "required": [ - "logLevel" - ], - "additionalProperties": false - }, - "serviceAccount": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "annotations": { - "$ref": "#/definitions/StringMap" - }, - "name": { - "type": "string" - } - }, - "required": [ - "annotations", - "create", - "name" - ], - "additionalProperties": false - }, - "watchNamespaces": { - "type": "array", - "items": { - "type": "string" - } - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - }, - "rbac": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - }, - "serviceMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "oneOf": [ - { - "properties": { - "watchNamespaces": { - "minItems": 1 - }, - "watchNamespaceSelector": { - "additionalProperties": false - } - } - }, - { - "properties": { - "watchNamespaces": { - "maxItems": 0 - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - } - } - } - ], - "required": [ - "affinity", - "config", - "image", - "updateStrategy", - "nodeSelector", - "podAnnotations", - "podLabels", - "rbac", - "replicaCount", - "resources", - "serviceAccount", - "serviceAnnotations", - "serviceLabels", - "serviceMonitor", - "tolerations" - ], - "additionalProperties": false - }, - "engine": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - }, - "sidecar": { - "type": "object", - "properties":{ - "podMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "required": [ - "podMonitor" - ], - "additionalProperties": false - } - }, - "required": [ - "image", - "resources", - "sidecar" - ], - "additionalProperties": false - }, - "networkValidator": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - } - }, - "required": [ - "image" - ], - "additionalProperties": false - }, - "sessionAgent": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - } - }, - "required": [ - "image", - "resources" - ], - "additionalProperties": false - }, - "license": { - "type": "object", - "properties": { - "secretName": { - "type": "string", - "minLength": 1 - } - }, - "required": [ - "secretName" - ], - "additionalProperties": false - }, - "dashboards": { - "type": "object", - "properties" : { "create": { "type": "boolean" }, - "config": { - "type": "object", - "properties": { - "grafana": { - "type": "object", - "properties": { - "folderAnnotation": { - "$ref": "#/definitions/NameValuePair" - }, - "dashboardLabel": { - "$ref": "#/definitions/NameValuePair" - } - }, - "required": [ - "folderAnnotation", - "dashboardLabel" - ], - "additionalProperties": false - } - }, - "required": [ - "grafana" - ], - "additionalProperties": false - }, - "instances": { - "type": "object", - "properties": { - "overview": { - "$ref": "#/definitions/DashboardInstance" - }, - "license" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockMetrics" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockLogs" : { - "$ref": "#/definitions/DashboardInstance" - } - }, - "required": [ - "overview", - "license", - "blockMetrics", - "blockLogs" - ], - "additionalProperties": false + "createSCCRole": { + "type": "boolean" } }, "required": [ "create", - "config", - "instances" + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" ], "additionalProperties": false }, @@ -357,18 +164,22 @@ } }, "required": [ + "affinity", "commonAnnotations", "commonLabels", - "crds", - "engine", + "config", "fullnameOverride", + "image", "imagePullSecrets", - "license", + "multusNetworkAttachmentDefinition", "nameOverride", - "operator", - "networkValidator", - "sessionAgent", - "dashboards", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", "tests" ], "additionalProperties": false, @@ -409,132 +220,6 @@ "tag" ], "additionalProperties": false - }, - "LabelSelector": { - "type": "object", - "properties": { - "matchExpressions": { - "type": "array", - "items": { - "type": "object", - "required": [ - "key", - "operator" - ], - "properties": { - "key": { - "type": "string" - }, - "operator": { - "type": "string" - }, - "values": { - "type": "array", - "items": { - "type": "string" - } - } - }, - "additionalProperties": false - } - }, - "matchLabels": { - "$ref": "#/definitions/StringMap" - } - }, - "additionalProperties": false - }, - "UpdateStrategy": { - "type": "object", - "oneOf" : [ - { - "properties": { - "type": { - "$ref": "#/definitions/RecreateType" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - }, - { - "properties": { - "type": { - "$ref": "#/definitions/RollingUpdateType" - }, - "rollingUpdate": { - "$ref": "#/definitions/RollingUpdate" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - } - ] - }, - "RecreateType": { - "type": "string", - "enum": [ - "Recreate" - ] - }, - "RollingUpdateType": { - "type": "string", - "enum": [ - "RollingUpdate" - ] - }, - "RollingUpdate": { - "type": "object", - "properties": { - "maxSurge": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - }, - "maxUnavailable": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - } - }, - "anyOf": [ - {"required": ["maxSurge"]}, - {"required": ["maxUnavailable"]} - ], - "additionalProperties": false - }, - "DashboardInstance" : { - "type" : "object", - "properties" : { - "create" : { - "type" : "boolean" - } - }, - "required" : [ - "create" - ], - "additionalProperties": false - }, - "NameValuePair" : { - "type" : "object", - "properties" : { - "name" : { - "type": "string", - "minLength": 1 - }, - "value" : { - "type" : "string", - "minLength": 1 - } - }, - "required" : [ - "name", - "value" - ], - "additionalProperties": false } } } diff --git a/charts/airlock/microgateway/4.3.3/values.yaml b/charts/airlock/microgateway/4.3.3/values.yaml index 03fc87d21..3dc707bae 100644 --- a/charts/airlock/microgateway/4.3.3/values.yaml +++ b/charts/airlock/microgateway/4.3.3/values.yaml @@ -1,4 +1,4 @@ -# -- Allows overriding the name to use instead of "microgateway". +# -- Allows overriding the name to use instead of "microgateway-cni". nameOverride: "" # -- Allows overriding the name to use as full name of resources. fullnameOverride: "" @@ -10,203 +10,75 @@ commonAnnotations: {} imagePullSecrets: [] # - name: myRegistryKeySecretName -crds: - # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. - # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster - # when performing a "helm install/upgrade". - skipVersionCheck: false -operator: - # -- Number of replicas for the operator Deployment. - replicaCount: 2 - # -- Specifies the operator update strategy. - updateStrategy: - type: RollingUpdate - # Specifies the Airlock Microgateway Operator image. - image: - # -- Image repository from which to pull the Airlock Microgateway Operator image. - repository: "quay.io/airlock/microgateway-operator" - # -- Image tag to pull. - tag: "4.3.3" - # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). - # Overrides tag when specified. - digest: "sha256:6d3ebca355de0a67f0bf5f088a15b9410564e500033d3e1f534a2f49a05bf4c3" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Annotations to add to all Pods. - podAnnotations: {} - # -- Labels to add to all Pods. - podLabels: {} - # -- Annotations to add to the Service. - serviceAnnotations: {} - # prometheus.io/scrape: "true" - # prometheus.io/port: "8080" - - # -- Labels to add to the Service. - serviceLabels: {} - # -- Resource restrictions to apply to the operator container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 1000m - # memory: 512Mi - # requests: - # cpu: 100m - # memory: 512Mi - - # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. - nodeSelector: {} - # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. - tolerations: [] - # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. - affinity: {} - # Parameters for the operator configuration. - config: - # -- Operator application log level. - logLevel: "info" - # Configures the generation of the ServiceAccount. - serviceAccount: - # -- Whether a ServiceAccount should be created. - create: true - # -- Annotations to add to the ServiceAccount. - annotations: {} - # -- Name of the ServiceAccount to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - # -- Allows to restrict the operator to specific namespaces, depending on your needs. - # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). - # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. - # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. - # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. - # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. - # Please note that this feature requires a Premium license. - watchNamespaces: [] - # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. - # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. - # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). - # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. - # Please note that this feature requires a Premium license. - watchNamespaceSelector: {} - # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. - # matchLabels: - # microgateway.airlock.com/enable: "true" - # matchExpressions: - # - { key: environment, operator: NotIn, values: [dev] } - - # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. - rbac: - # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. - create: true - # Configures the generation of a Prometheus Operator ServiceMonitor. - serviceMonitor: - # -- Whether to create a ServiceMonitor resource for monitoring. - create: false - # -- Labels to add to the ServiceMonitor. - labels: {} - # release: "" -engine: - # Specifies the Airlock Microgateway Engine image. - image: - # -- Image repository from which to pull the Airlock Microgateway Engine image. - repository: "quay.io/airlock/microgateway-engine" - # -- Image tag to pull. - tag: "4.3.3" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:3c0ebee0b560c8699723bfa433cd601b04b190c384e031d3789b83287fab7a9b" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Engine container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 500m - # memory: 128Mi - # requests: - # cpu: 10m - # memory: 40Mi - - # Additional configuration when deployed as a sidecar. - sidecar: - # Configures the generation of a Prometheus Operator PodMonitor. - podMonitor: - # -- Whether to create a PodMonitor resource for monitoring. - create: false - # -- Labels to add to the PodMonitor. - labels: {} - # release: "" -networkValidator: - # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. - image: - # -- Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. - repository: "cgr.dev/chainguard/netcat" - # -- Image tag to pull. - tag: "" - # -- SHA256 image digest to pull (in the format "sha256:6051975a14c51b9d3b525a06004d62a4d323c08ca58e3468343095a55a42fff2"). - # Overrides tag when specified. - digest: "sha256:6051975a14c51b9d3b525a06004d62a4d323c08ca58e3468343095a55a42fff2" - # -- Pull policy for this image. - pullPolicy: IfNotPresent -sessionAgent: - # Specifies the Airlock Microgateway Session Agent image. - image: - # -- Image repository from which to pull the Airlock Microgateway Session Agent image. - repository: "quay.io/airlock/microgateway-session-agent" - # -- Image tag to pull. - tag: "4.3.3" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:994bf4117adb74da4e05c22ffc168d9844bc68efa6a7fb96d73e849d1ef67b56" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 150m - # memory: 32Mi - # requests: - # cpu: 10m - # memory: 8Mi -license: - # -- Name of the secret containing the "microgateway-license.txt" key. - secretName: "airlock-microgateway-license" -# Creates dashboards in the form of ConfigMaps that can be imported -# by Grafana using its sidecar setup. -dashboards: - # -- Whether to create any ConfigMaps containing Grafana dashboards to import. +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.3.3" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:16317b9a8430059c15175673ad53e31d9e882a1d1af6576214eb1534d8ea6937" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. create: false - config: - # Configures the necessary label and annotations along with their values - # to enable Grafana to correctly identify the ConfigMaps containing - # dashboards and file them within a dedicated folder in the dashboard overview. - # These settings need to match the Grafana sidecar configuration. - grafana: - folderAnnotation: - # -- Name of the annotation containing the folder name to file dashboards into. - name: "grafana_folder" - # -- Name of the folder dashboards are filed into within the Grafana UI. - value: "Airlock Microgateway" - dashboardLabel: - # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. - name: "grafana_dashboard" - # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. - value: "1" - instances: - # Available dashboard instances that can be individually created/deployed. - overview: - # -- Whether to create the overview dashboard. - create: true - license: - # -- Whether to create the license dashboard. - create: true - blockMetrics: - # -- Whether to create the block metrics dashboard. - create: true - blockLogs: - # -- Whether to create the block logs dashboard. - create: true -# Check whether the installation of the Airlock Microgateway Helm Chart was successful. -# Requires a secret with a valid Airlock Microgateway license key already to be present. + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system tests: # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). # If set to false, `helm test` will not run any tests. diff --git a/charts/airlock/microgateway/4.3.4/.helmignore b/charts/airlock/microgateway/4.3.4/.helmignore index 101ff5ac5..8561d2892 100644 --- a/charts/airlock/microgateway/4.3.4/.helmignore +++ b/charts/airlock/microgateway/4.3.4/.helmignore @@ -21,8 +21,7 @@ .idea/ *.tmproj .vscode/ -# CRDs kustomization.yaml -/crds/kustomization.yaml + # Helm unit tests /tests /validation diff --git a/charts/airlock/microgateway/4.3.4/Chart.yaml b/charts/airlock/microgateway/4.3.4/Chart.yaml index afc603857..002c30798 100644 --- a/charts/airlock/microgateway/4.3.4/Chart.yaml +++ b/charts/airlock/microgateway/4.3.4/Chart.yaml @@ -9,15 +9,15 @@ annotations: - name: Airlock Microgateway Forum url: https://forum.airlock.com/ catalog.cattle.io/certified: partner - catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/display-name: Airlock Microgateway CNI catalog.cattle.io/kube-version: '>=1.25.0-0' - catalog.cattle.io/release-name: microgateway - charts.openshift.io/name: Airlock Microgateway + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI apiVersion: v2 appVersion: 4.3.4 -description: A Helm chart for deploying the Airlock Microgateway +description: A Helm chart for deploying the Airlock Microgateway CNI plugin home: https://www.airlock.com/en/microgateway -icon: file://assets/icons/microgateway.svg +icon: file://assets/icons/microgateway-cni.svg keywords: - WAF - Web Application Firewall @@ -30,14 +30,13 @@ keywords: - Filtering - DevSecOps - shift left -- control plane -- Operator +- CNI kubeVersion: '>=1.25.0-0' maintainers: - email: support@airlock.com name: Airlock url: https://www.airlock.com/ -name: microgateway +name: microgateway-cni sources: - https://github.com/airlock/microgateway type: application diff --git a/charts/airlock/microgateway/4.3.4/README.md b/charts/airlock/microgateway/4.3.4/README.md index 5028932b1..1559e00a4 100644 --- a/charts/airlock/microgateway/4.3.4/README.md +++ b/charts/airlock/microgateway/4.3.4/README.md @@ -1,4 +1,4 @@ -# Airlock Microgateway +# Airlock Microgateway CNI ![Version: 4.3.4](https://img.shields.io/badge/Version-4.3.4-informational?style=flat-square) ![AppVersion: 4.3.4](https://img.shields.io/badge/AppVersion-4.3.4-informational?style=flat-square) @@ -40,58 +40,43 @@ Check the official documentation at **[docs.airlock.com](https://docs.airlock.co The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. ## Prerequisites -* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) -* [Airlock Microgateway License](#obtain-airlock-microgateway-license) -* [cert-manager](https://cert-manager.io/) * [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) -In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. -For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. -### Obtain Airlock Microgateway License -1. Either request a community or premium license - * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) - * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) -2. Check your inbox and save the license file microgateway-license.txt locally. - -> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. -### Deploy cert-manager -```bash -helm repo add jetstack https://charts.jetstack.io -helm install cert-manager jetstack/cert-manager --version '1.15.1' -n cert-manager --create-namespace --set crds.enabled=true --wait -``` - -## Deploy Airlock Microgateway Operator - -> This guide assumes a microgateway-license.txt file is present in the working directory. - -1. Install CRDs and Operator. +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). ```bash - # Create namespace - kubectl create namespace airlock-microgateway-system - - # Install License - kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt - - # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) - helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.3.4' --wait + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.4/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.4/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). 2. (Recommended) You can verify the correctness of the installation with `helm test`. ```bash - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.4' - helm test airlock-microgateway -n airlock-microgateway-system --logs - helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.4' + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.4' ``` -### Upgrading CRDs - -The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. -CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: -```bash -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.3.4 --server-side --force-conflicts -``` - -**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. ## Support @@ -104,61 +89,33 @@ For the community edition, check our **[Airlock community forum](https://forum.a | Key | Type | Default | Description | |-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | | commonAnnotations | object | `{}` | Annotations to add to all resources. | | commonLabels | object | `{}` | Labels to add to all resources. | -| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | -| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | -| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | -| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | -| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | -| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | -| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | -| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | -| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | -| engine.image.digest | string | `"sha256:91e05c509bed3b51ff4888d7475980d56cbc85db121aa766d1bde413204f9070"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | -| engine.image.tag | string | `"4.3.4"` | Image tag to pull. | -| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | -| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | -| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | | fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:1e01310b3ad8566e9b39ee539ed5c959049aadda1a18c1a534e96d8865e20172"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.3.4"` | Image tag to pull. | | imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | -| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | -| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | -| networkValidator.image.digest | string | `"sha256:7a73d4b82a2d4165bbc5efa55de4fee9d43f2b1c1edb3505cdc8afd1361bad9b"` | SHA256 image digest to pull (in the format "sha256:7a73d4b82a2d4165bbc5efa55de4fee9d43f2b1c1edb3505cdc8afd1361bad9b"). Overrides tag when specified. | -| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| networkValidator.image.repository | string | `"cgr.dev/chainguard/netcat"` | Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. | -| networkValidator.image.tag | string | `""` | Image tag to pull. | -| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | -| operator.config.logLevel | string | `"info"` | Operator application log level. | -| operator.image.digest | string | `"sha256:6819c78d5570de66edce6c13964c6e1b4cc4746d0c0bc6f4975cd38e324828c0"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | -| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | -| operator.image.tag | string | `"4.3.4"` | Image tag to pull. | -| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | -| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | -| operator.podLabels | object | `{}` | Labels to add to all Pods. | -| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | -| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | -| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | -| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | -| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | -| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | -| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | -| operator.serviceLabels | object | `{}` | Labels to add to the Service. | -| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | -| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | -| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | -| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | -| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | -| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | -| sessionAgent.image.digest | string | `"sha256:df4e50d0929cb4c5e4486452979b59ec17f5e49a1516b685acd3a1ab0ddb3cf4"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | -| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | -| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | -| sessionAgent.image.tag | string | `"4.3.4"` | Image tag to pull. | -| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | | tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | ## License diff --git a/charts/airlock/microgateway/4.3.4/app-readme.md b/charts/airlock/microgateway/4.3.4/app-readme.md deleted file mode 100644 index e32cac025..000000000 --- a/charts/airlock/microgateway/4.3.4/app-readme.md +++ /dev/null @@ -1,28 +0,0 @@ -# Airlock Microgateway - -*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* - -## Features -* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. -* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction -* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication -* Content security filters for protecting against known attacks (OWASP Top 10) -* Access control to allow only authenticated users to access the protected services -* API security features like JSON parsing or OpenAPI specification enforcement - -For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. - -## Requirements -* [Airlock Microgateway CNI Helm Chart](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Also available as Rancher Chart) -* [Airlock Microgateway License](https://github.com/airlock/microgateway?tab=readme-ov-file#obtain-airlock-microgateway-license) (After obtaining the license install it according to the [documentation](https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-operator)) -* [cert-manager](https://cert-manager.io/docs/installation/) - -## Documentation and links - -Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. - -* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) -* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) -* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) -* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) -* [GitHub](https://github.com/airlock/microgateway) \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/accesscontrols.microgateway.airlock.com.yaml deleted file mode 100644 index 9dc81f14d..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/accesscontrols.microgateway.airlock.com.yaml +++ /dev/null @@ -1,124 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: accesscontrols.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: AccessControl - listKind: AccessControlList - plural: accesscontrols - singular: accesscontrol - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: AccessControl specifies the options to perform access control with a Microgateway Engine container. - 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: Specifies how the Airlock Microgateway Engine performs access control. - properties: - policies: - description: Policies configures access control policies. - items: - properties: - authorization: - description: Authorization configures how requests are authorized. An empty object value {} disables authorization. - properties: - authentication: - description: Authentication specifies that clients need to be authenticated with the provided method. - properties: - oidc: - description: OIDC configures client authentication using OpenID Connect. - properties: - oidcRelyingPartyRef: - description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - oidcRelyingPartyRef - type: object - type: object - type: object - identityPropagation: - description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. - properties: - actions: - description: Actions specifies the propagation actions. - items: - properties: - identityPropagationRef: - description: IdentityPropagationRef selects an IdentityPropagation to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - identityPropagationRef - type: object - type: array - onFailure: - description: |- - OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: - _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. - enum: - - Pass - type: string - required: - - actions - - onFailure - type: object - required: - - authorization - type: object - maxItems: 1 - minItems: 1 - type: array - required: - - policies - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/contentsecurities.microgateway.airlock.com.yaml deleted file mode 100644 index e63a5b1eb..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/contentsecurities.microgateway.airlock.com.yaml +++ /dev/null @@ -1,139 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: contentsecurities.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: ContentSecurity - listKind: ContentSecurityList - plural: contentsecurities - singular: contentsecurity - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: ContentSecurity specifies the options to secure an upstream web application with a Microgateway Engine container. - 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: Specifies the options to secure an upstream web application with a Microgateway Engine container. - properties: - apiProtection: - description: |- - APIProtection defines the relevant configurations to protect APIs. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - graphQLRef: - description: |- - GraphQLRef selects the relevant GraphQL configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - openAPIRef: - description: |- - OpenAPIRef selects the relevant OpenAPI configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - filter: - description: |- - Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests - to protect against various attack patterns. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - denyRulesRef: - description: |- - DenyRulesRef selects the relevant DenyRules configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - headerRewritesRef: - description: |- - HeaderRewritesRef selects the relevant HeaderRewrites. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - limitsRef: - description: |- - LimitsRef selects the relevant Limits configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - parserRef: - description: |- - ParserRef selects the relevant Parser configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/denyrules.microgateway.airlock.com.yaml deleted file mode 100644 index 7108ee5e0..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/denyrules.microgateway.airlock.com.yaml +++ /dev/null @@ -1,1804 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: denyrules.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: DenyRules - listKind: DenyRulesList - plural: denyrules - singular: denyrules - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - DenyRules configures request filtering using Airlock built-in and custom deny rules. - Deny rules establish a negative security model. They define prohibited patterns which, when a match is found in a request, lead to it being blocked from reaching the upstream web application. - To handle possible false positives, lower the security level or define fine-granular deny rule exceptions - If undefined, default settings are applied, designed to work with most upstream web application services. - 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: Specification of the desired deny rules behavior. - properties: - request: - description: Request configures deny rules for downstream requests. - properties: - builtIn: - description: BuiltIn configures the built-in deny rules. - properties: - exceptions: - description: Exceptions allows to define exceptions for specific requests and deny rules. - items: - description: |- - DenyRulesException defines an exception for deny rules. Exceptions may be defined by any or a combination of the following elements: blockedData (the request data causing a block) or requestConditions (properties of a request without taking into consideration the reason why a request has been blocked). - At least one of blockedData and requestConditions must be set. - properties: - blockedData: - description: BlockedData defines an exception based on the request data causing the block. - properties: - graphQL: - description: |- - GraphQL defines an exception based on a blocked GraphQL query. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header defines an exception based on a blocked header. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON defines an exception based on a blocked JSON property. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - jsonPath: - description: |- - JSONPath defines the JSONPath pattern to match the path within the JSON. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - key: - description: |- - Key defines the key of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of the JSON property. - At most one of key and value can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter defines an exception based on a blocked parameter. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source defines the source of the parameter. - enum: - - Query - - Post - - Any - type: string - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path defines an exception based on the blocked path. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment defines an exception based on a blocked path segment. - Only one of parameter, header, path, pathSegment, json or graphQL can be set. - properties: - segments: - description: Segments defines the position of a segment within the path. - properties: - index: - description: Index specifies an exact path segment position by index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value defines the value of a path segment. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - type: object - requestConditions: - description: RequestConditions defines an exception based on a property of a request without taking into consideration the reason why a request has been blocked. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKeys: - description: RuleKeys restricts the exception to a set of deny rules. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - type: object - type: array - overrides: - description: Overrides allows to override the builtIn settings for specific deny rules. - items: - description: DenyRulesOverride allows to override the builtIn settings for specific deny rules. - properties: - conditions: - description: Conditions select which built-in deny rules' settings will be adjusted. - properties: - ruleKeys: - description: RuleKeys is a list of built-in deny rule names. - items: - description: |- - A deny rule name can be any of the following values: - ENCODING | - EXPLOIT | - HPP | - HTML | - IDOR | - LDAP | - NOSQL | - OGNL | - PHP | - PROTOCOL | - SANITY | - SCANNING | - SQL | - TEMPLATE | - UNIXCMD | - WINCMD | - XSS - enum: - - ENCODING - - EXPLOIT - - HPP - - HTML - - IDOR - - LDAP - - NOSQL - - OGNL - - PHP - - PROTOCOL - - SANITY - - SCANNING - - SQL - - TEMPLATE - - UNIXCMD - - WINCMD - - XSS - type: string - minItems: 1 - type: array - types: - description: Types defines the type of attributes the override should be applied on. If Types are defined without any RuleKeys the override is applied to all deny rules. - items: - description: |- - A deny rule override type name can be any of the following values: - Header | - Parameter | - Path | - JSON | - GraphQL - enum: - - Header - - Parameter - - Path - - PathSegment - - JSON - - GraphQL - type: string - minItems: 0 - type: array - type: object - settings: - description: Settings override the corresponding properties for the selected rules. - properties: - level: - description: Level specifies the filter strength. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: array - settings: - description: Settings contains the keys which will be adjusted. - properties: - level: - default: Standard - description: Level represents a set of deny rules with different filter strengths. - enum: - - Unfiltered - - Basic - - Standard - - Strict - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - type: object - type: object - custom: - description: Custom allows configuring additional deny rules. - properties: - rules: - description: Rules defines list of additional deny rules. - items: - properties: - blockData: - description: BlockData specifies the request data which should cause a block. - properties: - graphQL: - description: |- - GraphQL specifies to block requests containing a matching GraphQL property. - At least one of field, argument and value must be set. - properties: - argument: - description: |- - Argument defines an argument of a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field defines a field of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value defines the value of an argument of the GraphQL query. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - header: - description: |- - Header specifies to block requests containing a matching header. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: |- - JSON specifies to block requests containing a matching JSON property in the body. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - key: - description: Key defines the key of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a JSON object. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - parameter: - description: |- - Parameter specifies to block requests containing a matching parameter. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - name: - description: Name defines the name of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a parameter. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - path: - description: |- - Path specifies to block requests with a matching path. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - matcher: - description: Matcher specifies which path to block. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - pathSegment: - description: |- - PathSegment specifies to block requests containing a matching path segment. - Only one of parameter, header, path, pathSegment or json can be set. - properties: - segments: - description: |- - Segments restricts which path segments are filtered by this rule. - If not specified, all segments of a path are filtered. - properties: - index: - description: Index restricts the rule to the path segment at this index (0-based). - minimum: 0 - type: integer - type: object - value: - description: Value specifies which path segment values to block. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - value - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - ruleKey: - description: RuleKey defines a technical key for the deny rule. Must be unique. - minLength: 1 - pattern: ^[A-Z][A-Z0-9_]*$ - type: string - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. - enum: - - Block - - LogOnly - type: string - required: - - blockData - - ruleKey - type: object - type: array - x-kubernetes-list-map-keys: - - ruleKey - x-kubernetes-list-type: map - type: object - type: object - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/envoyclusters.microgateway.airlock.com.yaml deleted file mode 100644 index 35dda9f2f..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/envoyclusters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: envoyclusters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyCluster - listKind: EnvoyClusterList - plural: envoyclusters - singular: envoycluster - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyCluster is an additional Envoy Cluster resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy cluster. - properties: - value: - description: Value defines the Envoy Cluster which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/envoyconfigurations.microgateway.airlock.com.yaml deleted file mode 100644 index c4f61f20d..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/envoyconfigurations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,185 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: envoyconfigurations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyConfiguration - listKind: EnvoyConfigurationList - plural: envoyconfigurations - singular: envoyconfiguration - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - EnvoyConfiguration is the Schema for the envoyconfigurations API - {{% notice warning %}} EnvoyConfiguration resources may contain sensitive information and thus RBAC permissions should be granted with care. {{% /notice %}} - 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: EnvoyConfigurationSpec defines the desired state of EnvoyConfiguration - properties: - envoyResources: - properties: - clusters: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - endpoints: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - extensions: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - listeners: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - routes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - runtimes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - scopedRoutes: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - secrets: - items: - type: object - x-kubernetes-preserve-unknown-fields: true - type: array - type: object - envoyResourcesRaw: - description: |- - EnvoyResourcesRaw defines the desired state for each resource type. The resources are stored as zstd compressed JSON bytes. - For debugging purposes, the resources can be inspected with the following command: `kubectl get envoyconfiguration -ojsonpath='{.spec.envoyResourcesRaw}' | base64 -d | zstd -d | jq` - format: byte - type: string - nodeID: - description: '**Deprecated:** This field is now ignored as NodeID is always derived from the resource name.' - type: string - type: object - status: - description: EnvoyConfigurationStatus defines the observed state of EnvoyConfiguration - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of EnvoyConfiguration condition. - type: string - required: - - status - - type - type: object - type: array - status: - type: string - xds: - properties: - resourceTypes: - additionalProperties: - description: XdsResourceTypeSyncStatus defines the sync status of xDS for a specific resource type - properties: - errorMessage: - description: ErrorMessage defines an optional message why the currently served resources of this resource type are rejected by the client. - type: string - resources: - additionalProperties: - description: XdsResourceStatus defines the status of xDS for a specific resource - properties: - version: - description: Version defines the version which is currently served for this resource. - type: string - required: - - version - type: object - description: Resources defines the resources which are currently served for this resource type. - type: object - status: - description: Status defines the current sync status of this resource type. - type: string - version: - description: Version defines the version which is currently served for this resource type. - type: string - required: - - resources - - status - - version - type: object - description: ResourceTypes defines the sync statuses for each resource type. - type: object - version: - description: Version defines the version of the underlying xDS snapshot. - type: integer - required: - - version - type: object - required: - - status - - xds - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/envoyhttpfilters.microgateway.airlock.com.yaml deleted file mode 100644 index 538ff672f..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/envoyhttpfilters.microgateway.airlock.com.yaml +++ /dev/null @@ -1,58 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: envoyhttpfilters.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: EnvoyHTTPFilter - listKind: EnvoyHTTPFilterList - plural: envoyhttpfilters - singular: envoyhttpfilter - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: EnvoyHTTPFilter is an additional Envoy HTTP Filter resource which is added to those defined by the Airlock Microgateway. - 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: Specification of the desired additional Envoy HTTP filter. - properties: - value: - description: Value defines the HTTP filter which is added to those configured by the Airlock Microgateway. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/graphqls.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/graphqls.microgateway.airlock.com.yaml deleted file mode 100644 index 165abe0a2..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/graphqls.microgateway.airlock.com.yaml +++ /dev/null @@ -1,88 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: graphqls.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: GraphQL - listKind: GraphQLList - plural: graphqls - singular: graphql - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: GraphQL contains the configuration for the GraphQL specification. - 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: Specification of the desired GraphQL specification. - properties: - settings: - description: Settings defines the settings to configure GraphQL. - properties: - allowIntrospection: - default: true - description: AllowIntrospection specifies if the introspection system is exposed. - type: boolean - allowMutations: - default: true - description: AllowMutations specifies if mutations are allowed. - type: boolean - schema: - description: Specifies the GraphQL schema. - properties: - source: - description: Source specifies the GraphQL schema to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'schema.graphql'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/headerrewrites.microgateway.airlock.com.yaml deleted file mode 100644 index 72a1067f9..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/headerrewrites.microgateway.airlock.com.yaml +++ /dev/null @@ -1,759 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: headerrewrites.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: HeaderRewrites - listKind: HeaderRewritesList - plural: headerrewrites - singular: headerrewrites - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. - properties: - request: - description: Request defines manipulations on upstream request headers. - properties: - add: - description: Add defines which request headers will be added before forwarding to the upstream. - properties: - custom: - description: |- - Custom allows configuring additional upstream request headers. - Add selected headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which request headers will be forwarded to the upstream. - This can either be allHeaders or matchingHeaders. - Default: matchingHeaders: {...} - properties: - allHeaders: - description: AllHeaders specifies that all request headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which request headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - standardHeaders: - default: true - description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which request headers will be removed before forwarding to the upstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream request headers. - properties: - alternativeForwardedHeaders: - default: true - description: |- - AlternativeForwardedHeaders removes downstream request headers which could potentially - be abused to alter the upstream's view of the remote connection. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream request headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - response: - description: Response defines manipulations on upstream response headers. - properties: - add: - description: Add defines which response headers will be added before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - csp: - default: true - description: |- - CSP sets a content security policy which allows only same-origin requests except for images - if the 'Content-Security-Policy' header is not set by the upstream. - type: boolean - featurePolicy: - default: false - description: |- - FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features - if the 'Feature-Policy' header is not set by the upstream. - **Deprecated:** Use permissionsPolicy instead. - type: boolean - hsts: - default: true - description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. - type: boolean - hstsPreload: - default: false - description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. - type: boolean - permissionsPolicy: - default: true - description: |- - PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features - if the 'Permissions-Policy' header is not set by the upstream. - type: boolean - referrerPolicy: - default: true - description: |- - ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests - if the 'Referrer-Policy' header is not set by the upstream. - type: boolean - xContentTypeOptions: - default: true - description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. - type: boolean - xFrameOptions: - default: true - description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to add. - items: - description: HeaderRewritesHeader specifies a header with a particular value - properties: - name: - description: Name defines the name of a header. - minLength: 1 - type: string - value: - description: Value defines the value of a header. - type: string - required: - - name - - value - type: object - minItems: 1 - type: array - mode: - default: AddIfAbsent - description: Mode defines the header addition strategy. - enum: - - AddIfAbsent - - OverwriteOrAdd - type: string - name: - description: Name describing the configured operation. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - allow: - description: |- - Allow defines which response headers will be forwarded to the downstream. - This can either be allHeaders or matchingHeaders. - Default: allHeaders: {} - properties: - allHeaders: - description: AllHeaders specifies that all response headers should be forwarded. - type: object - matchingHeaders: - description: MatchingHeaders specifies which response headers should be forwarded. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response header. - properties: - standardHeaders: - default: false - description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to allow. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - remove: - description: Remove defines which response headers will be removed before forwarding to the downstream. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined upstream response headers. - properties: - auth: - description: Auth defines the categories of headers concerning authentication. - properties: - basic: - default: false - description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. - type: boolean - negotiate: - default: true - description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. - type: boolean - ntlm: - default: true - description: |- - NTLM removes upstream response headers that advise clients to authenticate with NTLM. - By default, these headers are removed, because NTLM pass-through is not supported. - type: boolean - type: object - informationLeakage: - description: InformationLeakage defines the categories of headers concerning information leakage. - properties: - application: - default: true - description: Application removes upstream response headers that leak information about the deployed software. - type: boolean - server: - default: true - description: Server removes upstream response headers that leak information about the server. - type: boolean - type: object - permissiveCors: - default: true - description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. - type: boolean - type: object - custom: - description: Custom allows configuring additional upstream response headers. - items: - properties: - headers: - description: Headers to remove. - items: - description: |- - HeaderMatcher defines a matcher for an HTTP header. - At least one of name and value must be set. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - minItems: 1 - type: array - name: - description: Name describing the configured remove operation. Must be unique. - minLength: 1 - type: string - required: - - headers - - name - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - type: object - type: object - settings: - description: Settings configures the HeaderRewrites filter. - properties: - operationalMode: - default: Production - description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. - enum: - - Production - - Integration - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/identitypropagations.microgateway.airlock.com.yaml deleted file mode 100644 index 661e932f7..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/identitypropagations.microgateway.airlock.com.yaml +++ /dev/null @@ -1,108 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: identitypropagations.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: IdentityPropagation - listKind: IdentityPropagationList - plural: identitypropagations - singular: identitypropagation - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: IdentityPropagation specifies the desired identity propagation. - 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: Specification of the desired identity propagation. - properties: - header: - description: Header configures identity propagation via a request header. - properties: - name: - description: Name of the header to set. - minLength: 1 - type: string - value: - description: Value to propagate to the application. - properties: - source: - description: Source from which to extract the value. - properties: - metadata: - description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. - properties: - key: - description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. - minLength: 1 - type: string - namespace: - description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. - minLength: 1 - type: string - required: - - key - - namespace - type: object - oidc: - description: OIDC specifies to extract a value from the result of an OpenID Connect flow. - properties: - idToken: - description: IDToken specifies to extract the value from the OpenID Connect ID Token. - properties: - claim: - description: Claim selects the JWT claim from which to extract the value. - minLength: 1 - type: string - required: - - claim - type: object - required: - - idToken - type: object - type: object - required: - - source - type: object - required: - - name - - value - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/limits.microgateway.airlock.com.yaml deleted file mode 100644 index a75813dc4..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/limits.microgateway.airlock.com.yaml +++ /dev/null @@ -1,651 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: limits.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Limits - listKind: LimitsList - plural: limits - singular: limits - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Limits contains the configuration for limits. - 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: Specification of the desired limits behavior. - properties: - request: - description: Request defines the limits for requests. - properties: - limited: - description: Limited enables limits on request scope. - properties: - exceptions: - description: Exceptions defines limit exceptions. - items: - description: LimitsException defines an exception for limits. - properties: - length: - description: Length defines an exception for length limits based on the data element exceeding the limit. - properties: - graphQL: - description: GraphQL defines a field, argument or value length limit exception for a GraphQL query. - properties: - argument: - description: |- - Argument restricts the exception to GraphQL queries with a matching argument of a field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - field: - description: |- - Field restricts the exception to GraphQL queries with a matching field. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: |- - Value restricts the exception to GraphQL queries with a matching argument value. - At least one of field, argument and value must be set. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - json: - description: JSON defines a key and value length limit exception for a JSON property. - properties: - jsonPath: - description: |- - JSONPath restricts the exception to JSON properties with a matching JSONPath. - Expressions in JSONPath i.e. `?(expr)` are not supported. - minLength: 1 - type: string - required: - - jsonPath - type: object - parameter: - description: Parameter defines a name and value length limit exception for a parameter. - properties: - name: - description: Name restricts the exception to parameters with a matching name. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - source: - default: Any - description: Source restricts the exception to parameters of this kind. - enum: - - Query - - Post - - Any - type: string - required: - - name - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this exception to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - type: object - type: array - general: - description: General defines general request limits. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the request body. It specifies the number of bytes (0 = unlimited). This limit is effective for any request not processed by one of the content parsers (e.g. json) as configured in the Parser CRD. **Note** This limit does not apply to WebSocket or gRPC traffic. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - pathLength: - anyOf: - - type: integer - - type: string - default: 1Ki - description: PathLength defines the maximum path length for all requests (parsed and unparsed). - 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 - graphQL: - description: GraphQL defines the limits for GraphQL requests. - properties: - nestingDepth: - default: 10 - description: NestingDepth defines the maximum depth of nesting for GraphQL objects. - format: int64 - type: integer - querySize: - anyOf: - - type: integer - - type: string - default: 1Ki - description: QuerySize defines the maximum size for GraphQL queries. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: "256" - description: ValueLength defines the maximum length for GraphQL values. - 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 - json: - description: JSON defines the limits for JSON requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the JSON request body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - elementCount: - default: 10000 - description: ElementCount defines the maximum number of keys and array items in the whole JSON document (recursive). - format: int64 - type: integer - keyCount: - default: 250 - description: KeyCount defines the maximum number of keys of a single JSON object (non-recursive). - format: int64 - type: integer - keyLength: - anyOf: - - type: integer - - type: string - default: "128" - description: KeyLength defines the maximum length for JSON keys. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - nestingDepth: - default: 100 - description: NestingDepth defines the maximum depth of nesting for JSON objects and JSON arrays. - format: int64 - type: integer - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for JSON values. - 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 - multipart: - description: Multipart defines the limits for Multipart requests. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Mi - description: BodySize limits the total size of the Multipart request body. It specifies the number of bytes (0 = unlimited). - 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 - parameter: - description: Parameter defines the limits for request parameters. - properties: - bodySize: - anyOf: - - type: integer - - type: string - default: 100Ki - description: BodySize limits the total size of the form data body. It specifies the number of bytes (0 = unlimited). - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - count: - default: 128 - description: Count defines the maximum number of request parameters. - format: int64 - type: integer - nameLength: - anyOf: - - type: integer - - type: string - default: "128" - description: NameLength defines the maximum length for parameter names. - pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ - x-kubernetes-int-or-string: true - valueLength: - anyOf: - - type: integer - - type: string - default: 8Ki - description: ValueLength defines the maximum length for parameter values. - 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 - unlimited: - description: Unlimited disables all limits on request scope. - type: object - type: object - settings: - description: Settings configures the limits filter. - properties: - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled when a limit hits. - enum: - - Block - - LogOnly - type: string - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/oidcproviders.microgateway.airlock.com.yaml deleted file mode 100644 index 030bd153b..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/oidcproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,305 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: oidcproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCProvider - listKind: OIDCProviderList - plural: oidcproviders - singular: oidcprovider - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCProvider specifies an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - 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: Specification of an OpenID Provider. - properties: - static: - description: Static configures an OpenID Provider by explicitly specifying all endpoints. - properties: - endpoints: - description: Endpoints specifies the OpenID Provider endpoints. - properties: - authorization: - description: Authorization specifies the endpoint to which the authorization request is sent. - properties: - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - token: - description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. - properties: - tls: - description: TLS defines TLS settings. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: |- - Custom explicitly specifies how the server certificate should be verified. - Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines constraints the presented certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - disabled: - description: |- - Disabled specifies to trust any certificate without verification. - THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. - type: object - type: object - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - uri: - description: URI specifies the endpoint address. - format: uri - minLength: 1 - pattern: ^(http|https)://.*$ - type: string - required: - - uri - type: object - required: - - authorization - - token - type: object - required: - - endpoints - type: object - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/oidcrelyingparties.microgateway.airlock.com.yaml deleted file mode 100644 index 7398b262b..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/oidcrelyingparties.microgateway.airlock.com.yaml +++ /dev/null @@ -1,224 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: oidcrelyingparties.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OIDCRelyingParty - listKind: OIDCRelyingPartyList - plural: oidcrelyingparties - singular: oidcrelyingparty - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). - - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} - {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} - 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: Specification of the OIDC Relying Party configuration. - properties: - clientID: - description: ClientID specifies the OIDCRelyingParty "client_id". - minLength: 1 - type: string - credentials: - description: Credentials used for client authentication on the back-channel with the authorization server. - properties: - clientSecret: - description: ClientSecret authenticates with the client password issued by the OpenID Provider (OP). - properties: - method: - default: BasicAuth - description: Method specifies in which format the client secret is sent with the authorization request. - enum: - - BasicAuth - - FormURLEncoded - type: string - secretRef: - description: SecretRef specifies the kubernetes secret containing the client password with key "client.secret". - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - required: - - clientSecret - type: object - oidcProviderRef: - description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - pathMapping: - description: PathMapping configures the action matching. - properties: - logoutPath: - description: LogoutPath specifies which request paths should initiate a logout. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - redirectPath: - description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - required: - - logoutPath - - redirectPath - type: object - redirectURI: - description: |- - RedirectURI configures the "redirect_uri" parameter included in the authorization request. - May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. - minLength: 1 - type: string - required: - - clientID - - credentials - - oidcProviderRef - - pathMapping - - redirectURI - type: object - required: - - spec - type: object - served: true - storage: true - subresources: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/openapis.microgateway.airlock.com.yaml deleted file mode 100644 index b05f43ef2..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/openapis.microgateway.airlock.com.yaml +++ /dev/null @@ -1,167 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: openapis.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: OpenAPI - listKind: OpenAPIList - plural: openapis - singular: openapi - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: OpenAPI contains the configuration for the OpenAPI specification. - 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: Specification of the desired OpenAPI specification. - properties: - response: - description: Response defines the validation behaviour for responses. - properties: - secured: - description: Secured enables response checking. - properties: - validation: - default: Lax - description: Validation defines the validation mode for responses. - enum: - - Lax - - Strict - type: string - type: object - unsecured: - description: Unsecured disables response checking. - type: object - type: object - settings: - description: Settings defines the settings to configure OpenAPI specification enforcement. - properties: - logging: - description: Logging specifies the access log behavior. - properties: - maxFailedSubvalidations: - default: 10 - description: MaxFailedSubvalidations defines the maximum number of failed subvalidations being logged. - format: int64 - type: integer - type: object - schema: - description: Schema configures the OpenAPI specification. - properties: - source: - description: Source specifies the OpenAPI specification to be enforced. - properties: - configMapRef: - description: ConfigMapRef references the configmap by its name containing the well-known key 'openapi.json'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - required: - - source - type: object - threatHandlingMode: - default: Block - description: ThreatHandlingMode specifies how threats should be handled. - enum: - - Block - - LogOnly - type: string - validation: - description: Validation specifies the patterns for the validation behavior. - properties: - authentication: - description: Authentication defines the settings for the authentication scheme. - properties: - oAuth2: - description: OAuth2 specifies the OAuth2 parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - oidc: - description: Oidc specifies the OIDC parameters. - properties: - allowedParameters: - description: AllowedParameters specifies the allowed parameters for the authentication scheme. - properties: - builtIn: - description: BuiltIn allows configuring a set of predefined allowed parameters. - properties: - standardParameters: - default: true - description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. - type: boolean - type: object - custom: - description: Custom allows configuring additional allowed parameters. - items: - minLength: 1 - type: string - minItems: 1 - type: array - type: object - type: object - type: object - type: object - required: - - schema - type: object - required: - - settings - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/parsers.microgateway.airlock.com.yaml deleted file mode 100644 index 15171f2b7..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/parsers.microgateway.airlock.com.yaml +++ /dev/null @@ -1,358 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: parsers.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Parser - listKind: ParserList - plural: parsers - singular: parser - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Parser contains the configuration for content parsers (default and custom). - 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: Specification of the desired parser behavior. - properties: - request: - description: Request defines the parsing for downstream requests. - properties: - custom: - description: Custom allows configuring additional rules for parser selection. - properties: - rules: - description: |- - Rules defines a custom set prepended before built-in rules of enabled request parsers. - Disable all built-in parsers to overrule them completely. - items: - properties: - action: - description: |- - Action specifies what should happen when a request condition matches. - Only one of parse or skip can be set. - properties: - parse: - description: Parse activates the configured parser. - properties: - form: - description: Form activates the Form parser. - type: object - json: - description: JSON activates the JSON parser. - type: object - multipart: - description: Multipart activates the multipart parser. - type: object - type: object - skip: - description: Skip disables any content parsing - type: object - type: object - requestConditions: - description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. - properties: - header: - description: Header defines the matching headers of a request. - properties: - name: - description: Name defines the name of a header. - properties: - matcher: - description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - value: - description: Value defines the value of a header. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - type: object - invert: - default: false - description: Invert indicates whether the request condition should be inverted. - type: boolean - mediaType: - description: MediaType defines the matching media type from the content-type header of a request. - properties: - matcher: - description: |- - NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. - In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - method: - description: Method defines the matching methods of a request. - items: - description: Method defines common HTTP methods. - enum: - - GET - - HEAD - - POST - - PUT - - PATCH - - DELETE - - CONNECT - - OPTIONS - - TRACE - type: string - type: array - path: - description: Path defines the matching path of a request. - properties: - matcher: - description: StringMatcher defines the way to match a string. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - required: - - matcher - type: object - remoteIP: - description: RemoteIP defines the matching remote IPs of a request. - properties: - cidrRanges: - description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. - items: - description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. - format: cidr - type: string - minItems: 1 - type: array - invert: - default: false - description: Invert indicates whether the match should be inverted. - type: boolean - required: - - cidrRanges - type: object - type: object - required: - - action - - requestConditions - type: object - type: array - type: object - defaultContentType: - default: application/x-www-form-urlencoded - description: DefaultContentType specifies the content-type header which should be injected into the request before parser selection if it is not already present and the request has a body. - minLength: 1 - type: string - parsers: - description: Parsers defines the configuration for the available content parsers. - properties: - form: - description: Form defines the configuration for the form parser. - properties: - enable: - default: true - description: Enable defines whether form payloads are inspected. - type: boolean - mediaTypePattern: - default: .*urlencoded.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as form arguments. - minLength: 1 - type: string - type: object - json: - description: JSON defines the configuration for the JSON parser. - properties: - enable: - default: true - description: Enable defines whether json payloads are inspected. - type: boolean - mediaTypePattern: - default: .*json.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as JSON. - minLength: 1 - type: string - type: object - multipart: - description: Multipart defines the configuration for the multipart parser. - properties: - enable: - default: true - description: Enable defines whether multipart payloads are inspected. - type: boolean - mediaTypePattern: - default: .*multipart.* - description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as a multipart payload. - minLength: 1 - type: string - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/redisproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/redisproviders.microgateway.airlock.com.yaml deleted file mode 100644 index 9acdf4ddb..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/redisproviders.microgateway.airlock.com.yaml +++ /dev/null @@ -1,159 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: redisproviders.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: RedisProvider - listKind: RedisProviderList - plural: redisproviders - singular: redisprovider - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: RedisProvider contains a client configuration for connecting to a Redis database. - 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: Specification of a Redis database client configuration. - properties: - auth: - description: Auth specifies the Redis credentials. - properties: - password: - description: Password specifies the Redis password. - properties: - secretRef: - description: SecretRef selects the secret containing the Redis password under the key 'redis.password'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - username: - default: default - description: Username specifies the Redis username to authenticate with. - minLength: 1 - pattern: ^[^\s]+$ - type: string - required: - - password - type: object - mode: - description: Mode configures the redis deployment mode. - properties: - standalone: - description: Standalone specifies the standalone Redis instance to connect to. - properties: - host: - description: Host specifies the IP or hostname. - minLength: 1 - pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ - type: string - port: - default: 6379 - description: Port specifies the port. - maximum: 65535 - minimum: 1 - type: integer - required: - - host - type: object - type: object - timeouts: - description: Timeouts specifies the timeouts when interacting with the Redis endpoint. - properties: - connect: - default: 5s - description: Connect specifies the timeout for establishing a connection. - type: string - maxDuration: - default: 2s - description: MaxDuration specifies the response timeout. - type: string - type: object - tls: - description: TLS defines TLS settings. If not specified, TLS is disabled i.e. unencrypted TCP is used when connecting to the Redis instance. - properties: - certificateVerification: - description: CertificateVerification specifies how the certificate presented by the server is verified. - properties: - custom: - description: Custom explicitly specifies how the server certificate should be verified. - properties: - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - required: - - certificates - type: object - required: - - trustedCA - type: object - disabled: - description: 'Disabled specifies to trust any certificate without verification. THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. Note: This setting currently also disables TLS SNI.' - type: object - publicCAs: - description: PublicCAs specifies to only accept certificates with a SAN matching the host and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Session Agent’s base image. - type: object - type: object - type: object - required: - - mode - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/sessionhandlings.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/sessionhandlings.microgateway.airlock.com.yaml deleted file mode 100644 index bb4c0f9c1..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/sessionhandlings.microgateway.airlock.com.yaml +++ /dev/null @@ -1,77 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: sessionhandlings.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SessionHandling - listKind: SessionHandlingList - plural: sessionhandlings - singular: sessionhandling - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: |- - SessionHandling contains the configuration for session handling. - - - {{% notice warning %}} The Session Handling feature (required for OIDC) is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as high-availability Redis configurations (e.g. Sentinel/Cluster) are not yet supported. - {{% /notice %}} - 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: Specification of the desired session handling behavior. - properties: - persistence: - description: Persistence configures where to store the session state. - properties: - redisProviderRef: - description: RedisProviderRef specifies to cache session information in the provided Redis instance. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - redisProviderRef - type: object - required: - - persistence - type: object - required: - - spec - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/sidecargateways.microgateway.airlock.com.yaml deleted file mode 100644 index 6847f7393..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/sidecargateways.microgateway.airlock.com.yaml +++ /dev/null @@ -1,758 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: sidecargateways.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: SidecarGateway - listKind: SidecarGatewayList - plural: sidecargateways - singular: sidecargateway - scope: Namespaced - versions: - - additionalPrinterColumns: - - jsonPath: .status.status - name: Status - type: string - - jsonPath: .metadata.creationTimestamp - name: Age - type: date - name: v1alpha1 - schema: - openAPIV3Schema: - description: SidecarGateway contains the configuration how to configure the Airlock Microgateway Engine when used as Sidecar Container within the Pod of an application. - 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: Specification of the desired sidecar gateway behavior. - properties: - applications: - description: Applications defines applications which run on different ports. - items: - properties: - containerPort: - default: 8080 - description: |- - ContainerPort refers to the container port. - This must be a valid port number, 0 < x < 65536. - format: int32 - maximum: 65535 - minimum: 1 - type: integer - downstream: - description: Downstream defines the downstream configuration for this application - properties: - protocol: - description: |- - Protocol defines the exposed HTTP protocol version. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies that the protocol should be inferred. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies that the client is assumed to speak HTTP/1.1. - type: object - http2: - description: HTTP2 specifies that the client is assumed to speak HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - remoteIP: - description: |- - RemoteIP defines how the remote IP of a client is propagated. - Default: xff: {...} - properties: - connectionIP: - description: ConnectionIP configures to use the source IP address of the direct downstream connection. - type: object - customHeader: - description: CustomHeader specifies to use a custom header for remote IP extraction. - properties: - headerName: - description: HeaderName specifies the name of the custom header containing the remote IP. - minLength: 1 - type: string - required: - default: true - description: Required specifies if the custom header is required. If true and not available the request will be rejected with 403. - type: boolean - required: - - headerName - type: object - xff: - description: XFF configures to use the standard 'X-Forwarded-For' header for IP extraction. - properties: - numTrustedHops: - default: 1 - description: NumTrustedHops specifies to extract the client's originating IP from the nth rightmost entry in the X-Forwarded-For header. With the default value of 1, the IP is extracted from the rightmost entry. - format: int32 - minimum: 1 - type: integer - type: object - type: object - requestNormalizations: - description: RequestNormalizations defines a set of normalization actions which are applied to the request before route matching. - properties: - mergeSlashes: - default: true - description: MergeSlashes ensures that adjacent slashes in the path are merged into one. - type: boolean - normalizePath: - default: true - description: NormalizePath ensures normalization according to RFC 3986 without case normalization. - type: boolean - type: object - restrictions: - description: Restrictions defines restrictions for downstream. - properties: - http: - description: HTTP defines limits for the HTTP protocol. - properties: - headersLength: - anyOf: - - type: integer - - type: string - default: 60Ki - description: HeadersLength defines maximum size of all request headers combined. Requests that exceed this limit will receive a 431 response. - 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 - timeouts: - description: Timeouts defines timeouts for downstream - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - default: 5m - description: |- - Idle defines the settings for the idle timeout when no data is sent or received. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - maxDuration: - default: 5m - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - A value of 0 will completely disable the timeout. - Default: 5m - type: string - requestHeaders: - default: 10s - description: |- - RequestHeaders defines the duration before all request headers must be received. - A value of 0 will completely disable the timeout. - Default: 10s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - clientCertificate: - description: |- - ClientCertificate defines the TLS settings for verification of client certificates. - At most one of ignored, optional and required can be set. - Default: ignored: {} - properties: - ignored: - description: Ignored disables verification of the client certificate. - type: object - optional: - description: |- - Optional enables verification of the client certificate if one is presented. - In this mode only trustedCA and crl settings can be configured since certificatePinning and allowedSANs require a client certificate. - properties: - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - required: - - trustedCA - type: object - required: - description: |- - Required contains settings for client certificate verification. A client must present a valid certificate. - At least one of trustedCA and certificatePinning must be set. - properties: - allowedSANs: - description: |- - AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, - that is to say, the SAN is verified if at least one matcher is matched. - AllowedSANs requires trustedCA to be set. - items: - description: |- - TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the - Subject Alternative Name of the presented certificate matches one of the specified matchers. - properties: - matcher: - description: Matcher defines the string matcher for the SAN value. - properties: - contains: - description: |- - Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - exact: - description: |- - Exact defines an explicit match on the string specified here. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - ignoreCase: - default: false - description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. - type: boolean - prefix: - description: |- - Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - regex: - description: |- - Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. - The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - suffix: - description: |- - Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. - Only one of exact, prefix, suffix, regex or contains can be set. - minLength: 1 - type: string - type: object - sanType: - description: SanType defines the type of SAN matcher. - enum: - - DNS - - Email - - URI - - IPAddress - type: string - required: - - matcher - - sanType - type: object - minItems: 1 - type: array - certificatePinning: - description: |- - CertificatePinning defines the constraints a client certificate must fulfill. - If more than one constraint is configured only one must be satisfied. - At least one of allowedSPKIs and allowedHashes must be set. - properties: - allowedHashes: - description: |- - AllowedHashes is a list of hex-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - allowedSPKIs: - description: |- - AllowedSPKIs is a list of base64-encoded SHA-256 hashes. - If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. - items: - type: string - minItems: 1 - type: array - type: object - crl: - description: CRL defines the Certificate Revocation List (CRL) settings. - properties: - lists: - description: Lists defines the list of secretRefs containing Certificate Revocation Lists. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - validationMode: - default: VerifyChain - description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. - enum: - - VerifyLeafCertOnly - - VerifyChain - type: string - type: object - trustedCA: - description: TrustedCA defines which CA certificates are trusted. - properties: - certificates: - description: Certificates defines the list of secretRefs containing trusted CA certificates. - items: - properties: - secretRef: - description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - secretRef - type: object - minItems: 1 - type: array - verificationDepth: - default: 1 - description: |- - VerificationDepth specifies the hops in the certificate chain at which validation is performed. - 1 means that either the leaf or the signing CA must be in the set of trusted certificates. - format: int32 - type: integer - required: - - certificates - type: object - type: object - type: object - enable: - default: false - description: Enable defines if the downstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - secretRef: - description: SecretRef defines the reference to the TLS server certificate (secret of type kubernetes.io/tls). - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - xfcc: - description: |- - XFCC defines the handling of X-Forwarded-Client-Cert header. Meaning of the possible values: - _Sanitize_: Do not send the XFCC header to the next hop. This is the default value. - _ForwardOnly_: When the client connection is mTLS (Mutual TLS), forward the XFCC header in the request. - _AppendAndForward_: When the client connection is mTLS, append the client certificate information to the request’s XFCC header and forward it. - _SanitizeAndSet_: When the client connection is mTLS, reset the XFCC header with the client certificate information and send it to the next hop. - _AlwaysForwardOnly_: Always forward the XFCC header in the request, regardless of whether the client connection is mTLS. - Note: When forwarding the XFCC header in the request you might have to adjust the header length restrictions (See sidecargateway.spec.applications.downstream.restrictions.http) - enum: - - Sanitize - - ForwardOnly - - AppendAndForward - - SanitizeAndSet - - AlwaysForwardOnly - type: string - type: object - type: object - envoyHTTPFilterRefs: - description: EnvoyHTTPFilterRefs selects the relevant EnvoyHTTPFilters. - properties: - prepend: - description: Prepend selects the relevant EnvoyHTTPFilters which are added before those configured by the Airlock Microgateway. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - type: object - routes: - description: Routes defines the security configurations for different paths. The first matching route (from top to bottom) applies. - items: - description: |- - SidecarGatewayApplicationRoute defines the security configurations for different paths. - At most one of secured and unsecured can be set. - Default: secured: {...} - properties: - pathPrefix: - default: / - description: PathPrefix defines the path prefix used during route selection. - minLength: 1 - type: string - secured: - description: Secured enables WAF processing for this route. - properties: - accessControlRef: - description: |- - AccessControlRef selects the relevant AccessControl configuration resource. - If undefined, Airlock Microgateway does not perform any access control. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - contentSecurityRef: - description: |- - ContentSecurityRef selects the relevant ContentSecurity configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: object - unsecured: - description: |- - Unsecured disables all WAF functionality and therefore protection for this route. - WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. - type: object - type: object - type: array - x-kubernetes-list-map-keys: - - pathPrefix - x-kubernetes-list-type: map - telemetryRef: - description: |- - TelemetryRef selects the relevant Telemetry configuration resource. - If undefined, default settings are applied, designed to work with most upstream web application services. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - upstream: - description: Upstream defines the upstream configuration for this application - properties: - protocol: - description: |- - Protocol defines HTTP protocol version used to communicate with the upstream. At most one of http1, http2 and auto can be set. - Default: auto: {} - properties: - auto: - description: Auto specifies to negotiate the protocol with TLS ALPN (if TLS is enabled) or, as a fallback, use the same protocol that is used by the downstream connection. - properties: - http2: - description: HTTP2 specifies the settings for when HTTP/2 is inferred. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - http1: - description: HTTP1 specifies to use HTTP/1.1. - type: object - http2: - description: HTTP2 specifies to use HTTP/2. - properties: - allowConnect: - default: false - description: Allows proxying Websocket and other upgrades over H2 connect. - type: boolean - type: object - type: object - timeouts: - description: Timeouts defines the timeout settings. - properties: - http: - description: HTTP defines the settings for HTTP timeouts. - properties: - idle: - description: |- - Timeout defines the settings for http timeouts. If this setting is not specified, the value of applications[].downstream.timeouts.http.idle is inherited. - A value of 0 will completely disable the timeout. - type: string - maxDuration: - default: 15s - description: |- - MaxDuration defines the total duration for a HTTP request/response stream. - Default: 15s - type: string - type: object - type: object - tls: - description: TLS defines the TLS settings. - properties: - ciphers: - description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. - items: - type: string - minItems: 1 - type: array - enable: - default: false - description: Enable defines if the upstream connection is encrypted. - type: boolean - protocol: - description: Protocol defines the supported TLS protocol versions. - properties: - maximum: - description: Maximum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - minimum: - description: Minimum supported TLS version. - enum: - - TLSv1_0 - - TLSv1_1 - - TLSv1_2 - - TLSv1_3 - type: string - type: object - type: object - type: object - type: object - minItems: 1 - type: array - x-kubernetes-list-map-keys: - - containerPort - x-kubernetes-list-type: map - envoyClusterRefs: - description: EnvoyClusterRefs selects the relevant EnvoyClusters. - items: - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - type: array - x-kubernetes-list-map-keys: - - name - x-kubernetes-list-type: map - podSelector: - description: PodSelector defines to which Pods the configuration will be applied to. - properties: - matchLabels: - additionalProperties: - type: string - description: MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels. - type: object - type: object - sessionHandlingRef: - description: SessionHandlingRef selects the SessionHandling configuration to apply. - properties: - name: - description: Name of the resource - minLength: 1 - type: string - required: - - name - type: object - required: - - applications - type: object - status: - description: Most recently observed status of the SidecarGateway which is populated by the system. This data is read-only and may not be up to date. - properties: - conditions: - items: - properties: - lastTransitionTime: - description: Last time the condition transitioned from one status to another. - format: date-time - type: string - message: - description: A human-readable message indicating details about the transition. - type: string - reason: - description: The reason for the condition's last transition. - type: string - status: - description: Status of the condition, one of True, False, Unknown. - type: string - type: - description: Type of SidecarGateway condition. - type: string - required: - - status - - type - type: object - type: array - pods: - items: - properties: - envoyConfig: - description: EnvoyConfig indicates the name of the EnvoyConfig CR for the Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - status: - type: string - unmanagedPods: - items: - properties: - managedBy: - description: ManagedBy indicates the Airlock Microgateway Operator instance which manages this Pod. - type: string - name: - description: Name indicates the name of a Pod selected by the SidecarGateway. - type: string - sessionAgentSecret: - type: string - required: - - name - type: object - type: array - required: - - status - type: object - type: object - served: true - storage: true - subresources: - status: {} diff --git a/charts/airlock/microgateway/4.3.4/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.4/crds/telemetries.microgateway.airlock.com.yaml deleted file mode 100644 index d1a8897a7..000000000 --- a/charts/airlock/microgateway/4.3.4/crds/telemetries.microgateway.airlock.com.yaml +++ /dev/null @@ -1,96 +0,0 @@ -apiVersion: apiextensions.k8s.io/v1 -kind: CustomResourceDefinition -metadata: - annotations: - controller-gen.kubebuilder.io/version: v0.15.0 - labels: - app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.4 - name: telemetries.microgateway.airlock.com -spec: - group: microgateway.airlock.com - names: - categories: - - airlock-microgateway - kind: Telemetry - listKind: TelemetryList - plural: telemetries - singular: telemetry - scope: Namespaced - versions: - - name: v1alpha1 - schema: - openAPIV3Schema: - description: Telemetry contains the configuration for telemetry (logging, metrics & tracing). - 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: Specification of the desired telemetry behavior. - properties: - correlation: - description: Correlation defines the correlation aspects of Telemetry. - properties: - idSource: - description: IDSource specifies how an external correlation ID should be obtained for a request. If not specified, no correlation ID will be logged. - properties: - header: - description: Header specifies to extract the correlation ID from a request header. If the header is absent from a request, no correlation ID will be logged. - properties: - name: - default: X-Correlation-Id - description: Name of the header (case-insensitive) from which to extract the correlation ID. - minLength: 1 - type: string - type: object - required: - - header - type: object - request: - description: Request defines the request related correlation settings of Telemetry. - properties: - allowDownstreamRequestID: - default: true - description: AllowDownstreamRequestID defines whether trace sampling will consider a provided x-request-id. - type: boolean - alterRequestID: - default: true - description: AlterRequestID defines whether to alter the UUID to reflect the trace sampling decision. If disabled no modification to the UUID will be performed, this may break tracing in the upstream. - type: boolean - type: object - type: object - logging: - description: Logging defines the logging aspects of Telemetry. - properties: - accessLog: - description: AccessLog defines the access log settings of Telemetry. - properties: - format: - description: Format defines the Access Log format of the sidecar. - properties: - json: - description: JSON defines the Access Log format as JSON. - type: object - x-kubernetes-preserve-unknown-fields: true - type: object - type: object - type: object - type: object - type: object - served: true - storage: true diff --git a/charts/airlock/microgateway/4.3.4/dashboards/blockLogs.json b/charts/airlock/microgateway/4.3.4/dashboards/blockLogs.json deleted file mode 100644 index ef0ce6d62..000000000 --- a/charts/airlock/microgateway/4.3.4/dashboards/blockLogs.json +++ /dev/null @@ -1,510 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_LOKI", - "label": "Loki", - "description": "", - "type": "datasource", - "pluginId": "loki", - "pluginName": "Loki" - }, - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "loki", - "name": "Loki", - "version": "1.0.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "table", - "name": "Table", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "description": "Blocked requests by Airlock Microgateway retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for even a more granular filtering of the logs.", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "custom": { - "align": "auto", - "cellOptions": { - "type": "auto" - }, - "filterable": true, - "inspect": true - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "Namespace" - }, - "properties": [ - { - "id": "custom.width", - "value": 221 - }, - { - "id": "custom.filterable" - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Timestamp" - }, - "properties": [ - { - "id": "custom.width", - "value": 214 - }, - { - "id": "unit", - "value": "dateTimeAsIso" - }, - { - "id": "custom.filterable" - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Method" - }, - "properties": [ - { - "id": "custom.width", - "value": 89 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Client IP" - }, - "properties": [ - { - "id": "custom.width", - "value": 138 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Request ID" - }, - "properties": [ - { - "id": "custom.width", - "value": 328 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Block Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 116 - }, - { - "id": "custom.filterable", - "value": false - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Request Size" - }, - "properties": [ - { - "id": "custom.width", - "value": 126 - }, - { - "id": "unit", - "value": "bytes" - }, - { - "id": "custom.align", - "value": "right" - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Attack Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 217 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Application" - }, - "properties": [ - { - "id": "custom.width", - "value": 207 - } - ] - } - ] - }, - "gridPos": { - "h": 27, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 2, - "options": { - "cellHeight": "sm", - "footer": { - "countRows": false, - "enablePagination": true, - "fields": "", - "reducer": [ - "sum" - ], - "show": false - }, - "showHeader": true, - "sortBy": [] - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_deny_rule\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.deny_rules.matches\"\n| label_format block_type=\"deny_rules\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule_key }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Deny Rule Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_limit\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.limits.matches\"\n| label_format block_type=\"limits\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Limit Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_openapi\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.openapi.reference\", constraint=\"airlock.openapi.request.failed_validation.constraint\", position=\"airlock.openapi.request.failed_validation.position\", message=\"airlock.openapi.request.failed_validation.message\"\n| label_format block_type=\"openapi\", attack_type=\"openapi\", details=`{{.reference }}: {{.constraint }} at {{ .position }} ({{ .message }})` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "OpenAPI Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_parser\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", attack_type=\"airlock.parser\", failed_check=\"airlock.parser.matches[0].failed_check\", message=\"airlock.parser.matches[0].message\"\n| label_format block_type=\"parsing\", attack_type=\"parsing\", details=`{{.failed_check}}: {{.message}}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Parser Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_graphql\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.graphql.reference\", message=\"airlock.graphql.request.failed_validation.message\"\n| label_format block_type=\"graphql\", attack_type=\"graphql\", details=`{{ .reference }}: {{ .message }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "GraphQL Blocks" - } - ], - "title": "Blocked Request logs", - "transformations": [ - { - "id": "merge", - "options": {} - }, - { - "id": "extractFields", - "options": { - "format": "json", - "source": "labels" - } - }, - { - "id": "filterFieldsByName", - "options": { - "byVariable": false, - "include": { - "names": [ - "Time", - "attack_type", - "block_type", - "client_ip", - "details", - "http_method", - "namespace", - "request_id", - "request_size", - "url", - "pod" - ] - } - } - }, - { - "id": "organize", - "options": { - "excludeByName": { - "Line": true, - "id": true, - "labelTypes": true, - "labels": true, - "tsNs": false - }, - "includeByName": {}, - "indexByName": { - "Time": 0, - "attack_type": 7, - "block_type": 6, - "client_ip": 9, - "details": 8, - "http_method": 3, - "namespace": 1, - "pod": 2, - "request_id": 10, - "request_size": 5, - "url": 4 - }, - "renameByName": { - "Time": "Timestamp", - "attack_type": "Attack Type", - "block_type": "Block Type", - "client_ip": "Client IP", - "details": "Details", - "http_method": "Method", - "namespace": "Namespace", - "pod": "Pod", - "request_id": "Request ID", - "request_size": "Request Size", - "tsNs": "", - "url": "Path" - } - } - } - ], - "type": "table" - } - ], - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, - "hide": 2, - "includeAll": false, - "label": "DS_LOKI", - "multi": false, - "name": "DS_LOKI", - "options": [], - "query": "loki", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "hide": 0, - "includeAll": true, - "label": "Block Type", - "multi": true, - "name": "blockType", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - } - ] - }, - "time": { - "from": "now-15m", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway Blocked Request Logs", - "uid": "adnyzcvwnyadcc", - "version": 3, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/dashboards/blockMetrics.json b/charts/airlock/microgateway/4.3.4/dashboards/blockMetrics.json deleted file mode 100644 index ba383d22e..000000000 --- a/charts/airlock/microgateway/4.3.4/dashboards/blockMetrics.json +++ /dev/null @@ -1,758 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "panel", - "id": "barchart", - "name": "Bar chart", - "version": "" - }, - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "description": "Metrics on requests blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 6, - "title": "Airlock Microgateway Block Metrics", - "type": "row" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 1 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [ - { - "options": { - "match": "nan", - "result": { - "index": 0, - "text": "n/a" - } - }, - "type": "special" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 4, - "y": 1 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", - "fullMetaSearch": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Blocked Requests (%)", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "% Blocked Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "left", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "blue", - "value": null - } - ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] - }, - "gridPos": { - "h": 10, - "w": 20, - "x": 0, - "y": 5 - }, - "id": 3, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "timezone": [ - "" - ], - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "instant": false, - "legendFormat": "Requests per second", - "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "hide": false, - "instant": false, - "legendFormat": "% Blocks", - "range": true, - "refId": "Blocks" - } - ], - "title": "Requests vs. % Blocks", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Blocked requests by block type.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "super-light-orange", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisGridShow": true, - "axisLabel": "", - "axisPlacement": "auto", - "fillOpacity": 80, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineWidth": 0, - "scaleDistribution": { - "type": "linear" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "fieldMinMax": false, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "none" - }, - "overrides": [] - }, - "gridPos": { - "h": 11, - "w": 10, - "x": 0, - "y": 15 - }, - "id": 4, - "options": { - "barRadius": 0, - "barWidth": 0.8, - "fullHighlight": false, - "groupWidth": 0.7, - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "orientation": "horizontal", - "showValue": "never", - "stacking": "none", - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "asc" - }, - "xField": "block_type", - "xTickLabelRotation": 0, - "xTickLabelSpacing": 0 - }, - "pluginVersion": "10.4.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "instant": true, - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Block Type", - "transformations": [ - { - "id": "reduce", - "options": { - "includeTimeField": false, - "labelsToFields": true, - "mode": "seriesToRows", - "reducers": [ - "sum" - ] - } - } - ], - "type": "barchart" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Blocked requests by attack type, which are subsets of the various block types.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "light-orange", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "fillOpacity": 80, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "lineWidth": 1, - "scaleDistribution": { - "type": "linear" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 11, - "w": 10, - "x": 10, - "y": 15 - }, - "id": 5, - "options": { - "barRadius": 0, - "barWidth": 0.8, - "fullHighlight": false, - "groupWidth": 0.7, - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": false - }, - "orientation": "horizontal", - "showValue": "never", - "stacking": "none", - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - }, - "xField": "attack_type", - "xTickLabelRotation": 0, - "xTickLabelSpacing": 0 - }, - "pluginVersion": "10.4.3", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum by (attack_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "instant": true, - "legendFormat": "__auto", - "range": false, - "refId": "A" - } - ], - "title": "Attack Type", - "transformations": [ - { - "id": "reduce", - "options": { - "labelsToFields": true, - "reducers": [ - "sum" - ] - } - } - ], - "type": "barchart" - } - ], - "refresh": "", - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "Datasource Prometheus", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, - "hide": 2, - "includeAll": false, - "label": "DS_LOKI", - "multi": false, - "name": "DS_LOKI", - "options": [], - "query": "loki", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "hide": 0, - "includeAll": true, - "label": "Operator Namespace", - "multi": true, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": ".*", - "skipUrlSync": false, - "sort": 0, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "hide": 0, - "includeAll": true, - "label": "Block Type", - "multi": true, - "name": "blockType", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - } - ] - }, - "time": { - "from": "now-24h", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": { - "hidden": false - }, - "timezone": "browser", - "title": "Airlock Microgateway Block Metrics", - "uid": "ddnqoczu7qvb4cdd3dd", - "version": 3, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/dashboards/license.json b/charts/airlock/microgateway/4.3.4/dashboards/license.json deleted file mode 100644 index b9d5777e2..000000000 --- a/charts/airlock/microgateway/4.3.4/dashboards/license.json +++ /dev/null @@ -1,521 +0,0 @@ -{ - "__inputs": [ - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" - } - ], - "__elements": {}, - "__requires": [ - { - "type": "grafana", - "id": "grafana", - "name": "Grafana", - "version": "10.2.0" - }, - { - "type": "datasource", - "id": "prometheus", - "name": "Prometheus", - "version": "1.0.0" - }, - { - "type": "panel", - "id": "stat", - "name": "Stat", - "version": "" - }, - { - "type": "panel", - "id": "timeseries", - "name": "Time series", - "version": "" - } - ], - "annotations": { - "list": [ - { - "builtIn": 1, - "datasource": { - "type": "grafana", - "uid": "-- Grafana --" - }, - "enable": true, - "hide": true, - "iconColor": "rgba(0, 211, 255, 1)", - "name": "Annotations & Alerts", - "type": "dashboard" - } - ] - }, - "editable": true, - "fiscalYearStartMonth": 0, - "graphTooltip": 0, - "id": null, - "links": [ - { - "asDropdown": true, - "icon": "external link", - "includeVars": true, - "keepTime": true, - "tags": [ - "airlock-microgateway" - ], - "targetBlank": true, - "title": "Airlock Microgateway", - "tooltip": "", - "type": "dashboards", - "url": "" - } - ], - "panels": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [ - { - "options": { - "0": { - "color": "red", - "index": 1, - "text": "Invalid" - }, - "1": { - "color": "green", - "index": 0, - "text": "Valid" - } - }, - "type": "value" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 0 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "License Status", - "range": false, - "refId": "Licenses" - } - ], - "title": "License Status", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Expiry date of the Airlock Microgateway license associated with the selected operator.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "time: L" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 3, - "y": 0 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "min(microgateway_license_expiry_timestamp_seconds{namespace=~\"${operator_namespace.regex}\"})*1000", - "instant": true, - "legendFormat": "Expiry Date (MM/DD/YYYY)", - "range": false, - "refId": "A" - } - ], - "title": "License Expiry Date", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of licensed requests for applications protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 7, - "y": 0 - }, - "id": 6, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_license_max_rq_count_per_month{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Licensed Requests", - "range": false, - "refId": "A" - } - ], - "title": "Licensed Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Estimated number of requests protected by Airlock Microgateway over 30 days based on the last 7 days.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 5, - "x": 11, - "y": 0 - }, - "id": 2, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d]))/7*30", - "instant": true, - "legendFormat": "Estimated Requests", - "range": false, - "refId": "A" - } - ], - "title": "Requests over 30 days (estimated)", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Number of requests per week processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "blue", - "mode": "fixed" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 12, - "w": 16, - "x": 0, - "y": 4 - }, - "id": 5, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(avg_over_time(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d])[2m:30s]))", - "instant": false, - "legendFormat": "# Requests per week", - "range": true, - "refId": "A" - } - ], - "title": "Processed Requests per week", - "type": "timeseries" - } - ], - "schemaVersion": 39, - "tags": [ - "airlock-microgateway" - ], - "templating": { - "list": [ - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, - "hide": 2, - "includeAll": false, - "label": "DS_PROMETHEUS", - "multi": false, - "name": "DS_PROMETHEUS", - "options": [], - "query": "prometheus", - "refresh": 1, - "regex": "", - "skipUrlSync": false, - "type": "datasource" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_valid,namespace)", - "description": "", - "hide": 0, - "includeAll": false, - "label": "Operator Namespace", - "multi": false, - "name": "operator_namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 0, - "type": "query" - } - ] - }, - "time": { - "from": "now-7d", - "to": "now" - }, - "timeRangeUpdatedDuringEditOrView": false, - "timepicker": {}, - "timezone": "browser", - "title": "Airlock Microgateway License", - "uid": "cdpq79bzrr01se", - "version": 2, - "weekStart": "" -} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/gke-values.yaml b/charts/airlock/microgateway/4.3.4/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway/4.3.4/openshift-values.yaml b/charts/airlock/microgateway/4.3.4/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway/4.3.4/questions.yml b/charts/airlock/microgateway/4.3.4/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway/4.3.4/templates/NOTES.txt b/charts/airlock/microgateway/4.3.4/templates/NOTES.txt index 6e5ce218a..bb94ff521 100644 --- a/charts/airlock/microgateway/4.3.4/templates/NOTES.txt +++ b/charts/airlock/microgateway/4.3.4/templates/NOTES.txt @@ -1,47 +1,15 @@ -Thank you for installing Airlock Microgateway. +Thank you for installing Airlock Microgateway CNI. -Please ensure the following prerequisites are fulfilled: -* Cert-Manager is installed. - https://cert-manager.io/docs/installation/helm/ -* Airlock Microgateway CNI is also installed on the cluster. - https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni -* A valid Airlock Microgateway license is deployed in the Kubernetes secret 'airlock-microgateway-license'. - * Get a free Community license: https://airlock.com/en/microgateway-community - * Order a Premium license: https://airlock.com/en/microgateway-premium +Please ensure that the helm values'.config.cniNetDir' and '.config.cniBinDir' are configured for your Kubernetes distribution. +For further information, consider our manual https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }}. +The chapter 'Setup > Installation' describes how to set those settings correctly. Further information: -* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }} -* CRD API reference documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }}/api/crds +* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" . }} * Airlock Microgateway Labs: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=helm -{{- if .Values.crds.skipVersionCheck }} -Warning: CRD version check skipped -{{- else -}} -{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} -{{- if $outdatedCRDs -}} - {{- fail (printf ` - -Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. -Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: - -kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts - -If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` - .Chart.AppVersion) - -}} -{{- end -}} -{{- end -}} -{{- if .Values.tests.enabled -}} - {{- if .Values.operator.watchNamespaces -}} - {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) -}} - {{- fail (printf ` - -To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. -` - .Release.Namespace) - -}} - {{- end -}} - {{- end -}} -{{- end }} +Next steps: +* Install Airlock Microgateway (if not done already) + https://artifacthub.io/packages/helm/airlock-microgateway/microgateway Your release version is {{ .Chart.Version }}. \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/_helpers.tpl b/charts/airlock/microgateway/4.3.4/templates/_helpers.tpl index 733ba9648..996491a87 100644 --- a/charts/airlock/microgateway/4.3.4/templates/_helpers.tpl +++ b/charts/airlock/microgateway/4.3.4/templates/_helpers.tpl @@ -1,16 +1,14 @@ {{/* Expand the name of the chart. -We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest explicit suffix is 14 characters. */}} -{{- define "airlock-microgateway.name" -}} -{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Convert an image configuration object into an image ref string. */}} -{{- define "airlock-microgateway.image" -}} +{{- define "airlock-microgateway-cni.image" -}} {{- if .digest -}} {{- printf "%s@%s" .repository .digest -}} {{- else if .tag -}} @@ -22,19 +20,19 @@ Convert an image configuration object into an image ref string. {{/* Create a default fully qualified app name. -We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) -and the longest implicit suffix is 27 characters. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. If release name contains chart name it will be used as a full name. */}} -{{- define "airlock-microgateway.fullname" -}} +{{- define "airlock-microgateway-cni.fullname" -}} {{- if .Values.fullnameOverride }} -{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} {{- else }} {{- $name := default .Chart.Name .Values.nameOverride }} {{- if contains $name .Release.Name }} -{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} {{- else }} -{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} {{- end }} {{- end }} {{- end }} @@ -42,112 +40,62 @@ If release name contains chart name it will be used as a full name. {{/* Create chart name and version as used by the chart label. */}} -{{- define "airlock-microgateway.chart" -}} +{{- define "airlock-microgateway-cni.chart" -}} {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} {{- end }} {{/* Common labels */}} -{{- define "airlock-microgateway.sharedLabels" -}} -helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} {{- if .Chart.AppVersion }} app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} {{- end }} app.kubernetes.io/managed-by: {{ .Release.Service }} -app.kubernetes.io/part-of: {{ .Chart.Name }} {{- with .Values.commonLabels }} {{ toYaml .}} {{- end }} {{- end }} {{/* -Common Selector labels +Common labels without component */}} -{{- define "airlock-microgateway.sharedSelectorLabels" -}} -app.kubernetes.io/instance: {{ .Release.Name }} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} {{- end }} {{/* -Restricted Container Security Context +Selector labels */}} -{{- define "airlock-microgateway.restrictedSecurityContext" -}} -allowPrivilegeEscalation: false -privileged: false -runAsNonRoot: true -capabilities: - drop: ["ALL"] -readOnlyRootFilesystem: true -seccompProfile: - type: RuntimeDefault +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} {{- end }} -{{/* Precondition: May only be used if AppVersion is isSemver */}} -{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} -{{- $version := (semver .Chart.AppVersion) -}} -{{- if $version.Prerelease -}} ->= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} -{{- else -}} ->= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 -{{- end -}} -{{- end -}} +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} -{{- define "airlock-microgateway.outdatedCRDs" -}} -{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} - {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} - {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} - {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} - {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} - {{- $isOutdated := false -}} - {{- if $crd -}} - {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} - {{- $isOutdated = true -}} - {{- if hasKey $crd.metadata "labels" -}} - {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} - {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} - {{- if (semverCompare $supportedVersion $crdVersion) }} - {{- $isOutdated = false -}} - {{- end }} - {{- end -}} - {{- end -}} - {{- end -}} - {{- if $isOutdated }} -{{ base $path }} - {{- end }} - {{- end -}} -{{- end -}} -{{- end -}} - -{{- define "airlock-microgateway.isSemver" -}} +{{- define "airlock-microgateway-cni.isSemver" -}} {{- regexMatch `^(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-]+)*))?$` . -}} {{- end -}} -{{- define "airlock-microgateway.docsVersion" -}} -{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} {{- $version := (semver .Chart.AppVersion) -}} {{- $version.Major }}.{{ $version.Minor -}} {{- else -}} {{- print "latest" -}} {{- end -}} {{- end -}} - -{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} -{{- $list := list -}} -{{- with .matchLabels -}} - {{- range $key, $value := . -}} - {{- $list = append $list (printf "%s=%s" $key $value) -}} - {{- end -}} -{{- end -}} -{{- with .matchExpressions -}} - {{- range . -}} - {{- if has .operator (list "In" "NotIn") -}} - {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} - {{- else if eq .operator "Exists" -}} - {{- $list = append $list .key -}} - {{- else if eq .operator "DoesNotExist" -}} - {{- $list = append $list (printf "!%s" .key) -}} - {{- end -}} - {{- end -}} -{{- end -}} -{{- join "," $list -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/clusterrole.yaml b/charts/airlock/microgateway/4.3.4/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/clusterrolebinding.yaml b/charts/airlock/microgateway/4.3.4/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/configmap.yaml b/charts/airlock/microgateway/4.3.4/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway/4.3.4/templates/daemonset.yaml b/charts/airlock/microgateway/4.3.4/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway/4.3.4/templates/network-attachment-definition.yaml b/charts/airlock/microgateway/4.3.4/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.3.4/templates/operator/_operator_helpers.tpl deleted file mode 100644 index a540ff9f4..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/_operator_helpers.tpl +++ /dev/null @@ -1,42 +0,0 @@ -{{/* -Create a default fully qualified name for operator components. -*/}} -{{- define "airlock-microgateway.operator.fullname" -}} -{{ include "airlock-microgateway.fullname" . }}-operator -{{- end }} - - -{{/* -Common operator labels -*/}} -{{- define "airlock-microgateway.operator.labels" -}} -{{ include "airlock-microgateway.sharedLabels" . }} -{{ include "airlock-microgateway.operator.selectorLabels" . }} -{{- end }} - -{{/* -Operator Selector labels -*/}} -{{- define "airlock-microgateway.operator.selectorLabels" -}} -{{ include "airlock-microgateway.sharedSelectorLabels" . }} -app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-operator -app.kubernetes.io/component: controller -{{- end }} - -{{/* -Create the name of the service account to use for the operator -*/}} -{{- define "airlock-microgateway.operator.serviceAccountName" -}} -{{- if .Values.operator.serviceAccount.create }} -{{- default (include "airlock-microgateway.operator.fullname" .) .Values.operator.serviceAccount.name }} -{{- else }} -{{- default "default" .Values.operator.serviceAccount.name }} -{{- end }} -{{- end }} - -{{/* -ServiceMonitor metrics regex pattern for leader only metrics -*/}} -{{- define "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" -}} -^(microgateway_license|microgateway_sidecars).*$ -{{- end }} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.3.4/templates/operator/_rbac.gen.tpl deleted file mode 100644 index 83b314cbc..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/_rbac.gen.tpl +++ /dev/null @@ -1,237 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator rbac permission rules -*/}} -{{- define "airlock-microgateway-operator.rbacRules" -}} -- apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list - - watch -- apiGroups: - - "" - resources: - - pods - verbs: - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - "" - resources: - - pods/finalizers - verbs: - - update -- apiGroups: - - "" - resources: - - pods/status - verbs: - - patch - - update -- apiGroups: - - "" - resources: - - secrets - verbs: - - create - - delete - - get - - list - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - accesscontrols - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - contentsecurities - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - denyrules - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyclusters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations - verbs: - - create - - delete - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - envoyconfigurations/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - envoyhttpfilters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - graphqls - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - headerrewrites - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - identitypropagations - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - limits - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcrelyingparties - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - openapis - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - parsers - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - redisproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sessionhandlings - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways - verbs: - - get - - list - - patch - - update - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/finalizers - verbs: - - update -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - telemetries - verbs: - - get - - list - - watch -{{- end }} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.3.4/templates/operator/_webhooks.gen.tpl deleted file mode 100644 index 02e304890..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/_webhooks.gen.tpl +++ /dev/null @@ -1,339 +0,0 @@ -{{/* AUTOGENERATED FILE DO NOT EDIT */}} - -{{/* -Operator mutating webhooks -*/}} -{{- define "airlock-microgateway-operator.mutatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /mutate-v1-pod - failurePolicy: Fail - name: mutate-pod.microgateway.airlock.com - reinvocationPolicy: IfNeeded - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -{{- end }} - -{{/* -Operator validating webhooks -*/}} -{{- define "airlock-microgateway-operator.validatingWebhooks" -}} -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-v1-pod - failurePolicy: Fail - name: validate-pod.microgateway.airlock.com - rules: - - apiGroups: - - "" - apiVersions: - - v1 - operations: - - CREATE - - UPDATE - resources: - - pods - sideEffects: None - objectSelector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-accesscontrol - failurePolicy: Fail - name: validate-accesscontrol.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - accesscontrols - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-denyrules - failurePolicy: Fail - name: validate-denyrules.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - denyrules - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoycluster - failurePolicy: Fail - name: validate-envoycluster.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyclusters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-envoyhttpfilter - failurePolicy: Fail - name: validate-envoyhttpfilter.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - envoyhttpfilters - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-graphql - failurePolicy: Fail - name: validate-graphql.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - graphqls - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-headerrewrites - failurePolicy: Fail - name: validate-headerrewrites.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - headerrewrites - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-identitypropagation - failurePolicy: Fail - name: validate-identitypropagation.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - identitypropagations - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-limits - failurePolicy: Fail - name: validate-limits.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - limits - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcprovider - failurePolicy: Fail - name: validate-oidcprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-oidcrelyingparty - failurePolicy: Fail - name: validate-oidcrelyingparty.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - oidcrelyingparties - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-openapi - failurePolicy: Fail - name: validate-openapi.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - openapis - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-parser - failurePolicy: Fail - name: validate-parser.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - parsers - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-redisprovider - failurePolicy: Fail - name: validate-redisprovider.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - redisproviders - sideEffects: None -- admissionReviewVersions: - - v1 - clientConfig: - service: - name: airlock-microgateway-operator-webhook - namespace: '{{ .Release.Namespace }}' - path: /validate-microgateway-airlock-com-v1alpha1-sidecargateway - failurePolicy: Fail - name: validate-sidecargateway.microgateway.airlock.com - rules: - - apiGroups: - - microgateway.airlock.com - apiVersions: - - v1alpha1 - operations: - - CREATE - - UPDATE - resources: - - sidecargateways - sideEffects: None -{{- end }} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/configmap.yaml deleted file mode 100644 index 95e52d7df..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/configmap.yaml +++ /dev/null @@ -1,394 +0,0 @@ -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -data: - engine_bootstrap_config_template.yaml: | - # Base configuration, admin interface on port 19000 - admin: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - dynamic_resources: - cds_config: - initial_fetch_timeout: 10s - resource_api_version: V3 - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - lds_config: - resource_api_version: V3 - initial_fetch_timeout: 10s - api_config_source: - api_type: GRPC - transport_api_version: V3 - grpc_services: - - envoy_grpc: - cluster_name: xds_cluster - set_node_on_first_message_only: true - # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC - rate_limit_settings: - max_tokens: 5 - fill_rate: 0.2 - static_resources: - listeners: - - name: probe - address: - socket_address: - address: 0.0.0.0 - port_value: 19001 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: probe - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: probe - virtual_hosts: - - name: probe - domains: - - '*' - routes: - - name: ready - match: - path: /ready - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - - name: metrics - address: - socket_address: - address: 0.0.0.0 - port_value: 19002 - filter_chains: - - filters: - - name: http_connection_manager - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager - stat_prefix: metrics - codec_type: AUTO - http2_protocol_options: - initial_connection_window_size: 1048576 - initial_stream_window_size: 65536 - max_concurrent_streams: 100 - route_config: - name: metrics - virtual_hosts: - - name: metrics - domains: - - '*' - routes: - - name: metrics - match: - path: /metrics - headers: - - name: ':method' - string_match: - exact: 'GET' - route: - prefix_rewrite: '/stats/prometheus' - cluster: airlock_microgateway_engine_admin - http_filters: - - name: envoy.filters.http.router - typed_config: - '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router - clusters: - - name: xds_cluster - connect_timeout: 1s - type: STRICT_DNS - load_assignment: - cluster_name: xds_cluster - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: airlock-microgateway-operator-xds.$(OPERATOR_NAMESPACE).svc.cluster.local - port_value: 13377 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - transport_socket: - name: envoy.transport_sockets.tls - typed_config: - '@type': type.googleapis.com/envoy.extensions.transport_sockets.tls.v3.UpstreamTlsContext - common_tls_context: - tls_params: - tls_minimum_protocol_version: TLSv1_3 - tls_maximum_protocol_version: TLSv1_3 - validation_context_sds_secret_config: - name: validation_context_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/validation_context_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - tls_certificate_sds_secret_configs: - - name: tls_certificate_sds - sds_config: - resource_api_version: V3 - path_config_source: - path: /etc/envoy/tls_certificate_sds_secret.yaml - watched_directory: - path: /etc/envoy/ - - name: airlock_microgateway_engine_admin - connect_timeout: 1s - type: STATIC - load_assignment: - cluster_name: airlock_microgateway_engine_admin - endpoints: - - lb_endpoints: - - endpoint: - address: - socket_address: - address: 127.0.0.1 - port_value: 19000 - typed_extension_protocol_options: - envoy.extensions.upstreams.http.v3.HttpProtocolOptions: - '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions - explicit_http_config: - http2_protocol_options: - connection_keepalive: - interval: 360s - timeout: 5s - stats_config: - stats_tags: - - tag_name: "block_type" - regex: "\\.(block_type\\.([^.]+))" - - tag_name: "attack_type" - regex: "\\.(attack_type\\.([^.]+))" - - tag_name: "envoy_cluster_name" - regex: "\\.(cluster\\.([^.]+))" - - tag_name: "version" - regex: "\\.(version\\.([^.]+))" - use_all_default_tags: true - overload_manager: - resource_monitors: - - name: "envoy.resource_monitors.global_downstream_max_connections" - typed_config: - "@type": type.googleapis.com/envoy.extensions.resource_monitors.downstream_connections.v3.DownstreamConnectionsConfig - max_active_downstream_connections: 50000 - bootstrap_extensions: - - name: airlock.bootstrap.engine_build_info - typed_config: - '@type': type.googleapis.com/airlock.extensions.bootstrap.stats.v1alpha.Stats - application_log_config: - log_format: - text_format: '{"@timestamp":"%Y-%m-%dT%T.%e%z","log":{"logger":"%n","level":"%l","origin":{"file":{"name":"%g","line":%#},"function":"%!"}},"event":{"module":"envoy","dataset":"envoy.application"},"process":{"pid":%P,"thread":{"id":%t}},"ecs":{"version":"8.5"},"message":"%j"}' - engine_container_template.yaml: | - name: "$(ENGINE_NAME)" - image: "$(ENGINE_IMAGE)" - imagePullPolicy: {{ .Values.engine.image.pullPolicy }} - args: - - "--config-path" - - "/etc/envoy/bootstrap_config.yaml" - - "--base-id" - - "$(BASE_ID)" - - "--file-flush-interval-msec" - - '1000' - - "--drain-time-s" - - '60' - - "--service-node" - - "$(POD_NAME).$(POD_NAMESPACE)" - - "--service-cluster" - - "$(APP_NAME).$(POD_NAMESPACE)" - - "--log-path" - - "/dev/stdout" - - "--log-level" - - "$(LOG_LEVEL)" - volumeMounts: - - name: airlock-microgateway-bootstrap-secret-volume - mountPath: /etc/envoy - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - - name: POD_IP - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: status.podIP - ports: - - containerPort: 13378 - protocol: TCP - - containerPort: 19001 - protocol: TCP - - containerPort: 19002 - protocol: TCP - livenessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 1 - readinessProbe: - httpGet: - path: /ready - port: 19001 - scheme: HTTP - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.engine.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - session_agent_container_template.yaml: | - name: "$(SESSION_AGENT_NAME)" - image: "$(SESSION_AGENT_IMAGE)" - imagePullPolicy: {{ .Values.sessionAgent.image.pullPolicy }} - args: - - "--port" - - "19004" - - "--config-path" - - "/etc/microgateway-session-agent/config.json" - volumeMounts: - - name: airlock-microgateway-session-agent-volume - mountPath: /etc/microgateway-session-agent - readOnly: true - env: - - name: POD_NAME - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.name - - name: POD_NAMESPACE - valueFrom: - fieldRef: - apiVersion: v1 - fieldPath: metadata.namespace - ports: - - containerPort: 19004 - livenessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 5 - successThreshold: 1 - timeoutSeconds: 5 - readinessProbe: - {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} - grpc: - port: 19004 - {{- else }} - tcpSocket: - port: 19004 - {{- end }} - initialDelaySeconds: 5 - periodSeconds: 5 - failureThreshold: 3 - successThreshold: 1 - timeoutSeconds: 5 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - {{- with .Values.sessionAgent.resources }} - resources: - {{- toYaml . | nindent 6 }} - {{- end }} - network_validator_container_template.yaml: | - name: "$(NETWORK_VALIDATOR_NAME)" - image: "$(NETWORK_VALIDATOR_IMAGE)" - imagePullPolicy: {{ .Values.networkValidator.image.pullPolicy }} - command: ["/bin/sh", "-c"] - args: - - |- - echo 'pong' | nc -v -l 127.0.0.1 13378 & - for i in 1 2 3; do - sleep 1s - if r=$(echo 'ping' | nc -v -q 0 127.0.0.1 19003) && [ $r == pong ]; then - echo -n 'Traffic redirection to Airlock Microgateway Engine is working.' > /dev/termination-log - exit 0 - fi - done - echo -en 'Traffic redirection to Airlock Microgateway Engine is not working.\nRestart the pod after ensuring that hostNetwork is disabled and a compatible Airlock Microgateway CNI version is installed on the node.\nCertain environments may also require additional configuration (see docs.airlock.com for more information).' > /dev/termination-log - exit 1 - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} - runAsUser: $(SECURITYCONTEXT_UID) - operator_config.yaml: | - apiVersion: config.airlock.com/v1alpha1 - kind: OperatorConfig - health: - healthProbeBindAddress: :8081 - metrics: - bindAddress: 0.0.0.0:8080 - webhook: - port: 9443 - deployment: - sidecar: - engineContainerTemplate: "/sidecar/engine_container_template.yaml" - networkValidatorContainerTemplate: "/sidecar/network_validator_container_template.yaml" - sessionAgentContainerTemplate: "/sidecar/session_agent_container_template.yaml" - engine: - bootstrapConfigTemplate: "/engine_bootstrap_config_template.yaml" - log: - level: {{ .Values.operator.config.logLevel }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaces: - selector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaces: - list: - {{- toYaml . | nindent 8 }} - {{- end }} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/dashboard-configmap.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/dashboard-configmap.yaml deleted file mode 100644 index b71ac89b6..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/dashboard-configmap.yaml +++ /dev/null @@ -1,28 +0,0 @@ -{{- if .Values.dashboards.create -}} -{{- range $instance := (keys .Values.dashboards.instances | sortAlpha) -}} -{{- $dashboard := get $.Values.dashboards.instances $instance -}} -{{- if $dashboard.create }} ---- -apiVersion: v1 -kind: ConfigMap -metadata: - name: {{ include "airlock-microgateway.fullname" $ }}-dashboard-{{ $instance | lower }} - namespace: {{ $.Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.dashboards.config.grafana.dashboardLabel -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - annotations: - {{- with $.Values.dashboards.config.grafana.folderAnnotation -}} - {{- .name | nindent 4 -}}: {{ .value | quote }} - {{- end }} - {{- with $.Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -data: - {{- printf "%s.json" $instance | nindent 2 }}: |- - {{- ($.Files.Get (printf "dashboards/%s.json" $instance)) | nindent 4 -}} -{{- end -}} -{{- end -}} -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/deployment.yaml deleted file mode 100644 index db340cdec..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/deployment.yaml +++ /dev/null @@ -1,143 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - replicas: {{ .Values.operator.replicaCount }} - {{- with .Values.operator.updateStrategy }} - strategy: - {{- toYaml . | trim | nindent 4 }} - {{- end }} - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - checksum/config: {{ include (print $.Template.BasePath "/operator/configmap.yaml") . | sha256sum }} - kubectl.kubernetes.io/default-container: manager - {{- with mustMerge .Values.operator.podAnnotations .Values.commonAnnotations}} - {{- toYaml . | nindent 8 }} - {{- end }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 8 }} - {{- with .Values.operator.podLabels }} - {{- toYaml . | nindent 8 }} - {{- end }} - spec: - containers: - - args: - - --config=operator_config.yaml - env: - - name: ENGINE_IMAGE - value: {{ include "airlock-microgateway.image" .Values.engine.image }} - - name: NETWORK_VALIDATOR_IMAGE - value: {{ include "airlock-microgateway.image" .Values.networkValidator.image }} - - name: SESSION_AGENT_IMAGE - value: {{ include "airlock-microgateway.image" .Values.sessionAgent.image }} - - name: OPERATOR_NAME - valueFrom: - fieldRef: - fieldPath: metadata.name - - name: OPERATOR_NAMESPACE - valueFrom: - fieldRef: - fieldPath: metadata.namespace - image: {{ include "airlock-microgateway.image" .Values.operator.image }} - imagePullPolicy: {{ .Values.operator.image.pullPolicy }} - livenessProbe: - httpGet: - path: /healthz - port: 8081 - initialDelaySeconds: 15 - periodSeconds: 20 - timeoutSeconds: 5 - name: manager - ports: - - containerPort: 9443 - name: webhook-server - protocol: TCP - - containerPort: 13377 - name: xds-server - protocol: TCP - - containerPort: 8080 - protocol: TCP - - containerPort: 8081 - protocol: TCP - readinessProbe: - httpGet: - path: /readyz - port: 8081 - initialDelaySeconds: 5 - periodSeconds: 10 - timeoutSeconds: 5 - {{- with .Values.operator.resources }} - resources: - {{- toYaml . | nindent 10 }} - {{- end }} - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 10 }} - terminationMessagePath: /dev/termination-log - terminationMessagePolicy: FallbackToLogsOnError - volumeMounts: - - mountPath: /tmp/k8s-webhook-server/serving-certs - name: cert - readOnly: true - - mountPath: /opt/airlock/license/ - name: airlock-microgateway-license - readOnly: true - - mountPath: /operator_config.yaml - name: operator-config - subPath: operator_config.yaml - - mountPath: /sidecar/engine_container_template.yaml - name: operator-config - subPath: engine_container_template.yaml - - mountPath: /sidecar/network_validator_container_template.yaml - name: operator-config - subPath: network_validator_container_template.yaml - - mountPath: /sidecar/session_agent_container_template.yaml - name: operator-config - subPath: session_agent_container_template.yaml - - mountPath: /engine_bootstrap_config_template.yaml - name: operator-config - subPath: engine_bootstrap_config_template.yaml - securityContext: - runAsNonRoot: true - serviceAccountName: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - terminationGracePeriodSeconds: 10 - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.nodeSelector }} - nodeSelector: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.tolerations }} - tolerations: - {{- toYaml . | nindent 8 }} - {{- end }} - {{- with .Values.operator.affinity }} - affinity: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - name: cert - secret: - defaultMode: 420 - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert - - name: airlock-microgateway-license - secret: - defaultMode: 292 - optional: true - secretName: {{ .Values.license.secretName }} - - configMap: - name: {{ include "airlock-microgateway.operator.fullname" . }}-config - name: operator-config diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/manager-role.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/manager-role.yaml deleted file mode 100644 index 90335bcfe..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/manager-role.yaml +++ /dev/null @@ -1,33 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" . -}} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: -{{ include "airlock-microgateway-operator.rbacRules" $ }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/manager-rolebinding.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/manager-rolebinding.yaml deleted file mode 100644 index ae99cfb7b..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/manager-rolebinding.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create }} -{{- if empty .Values.operator.watchNamespaces }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} -{{- else }} -{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager - namespace: {{ $namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} - {{- with $.Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" $ }} - namespace: {{ $.Release.Namespace }} ---- -{{- end -}} -{{- end -}} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/metrics-service.yaml deleted file mode 100644 index 34d23f6d6..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/metrics-service.yaml +++ /dev/null @@ -1,47 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} ---- -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-leader-metrics - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - operator.microgateway.airlock.com/isLeader: "true" - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: http - name: metrics - port: 8080 - protocol: TCP - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/mutating-webhook.yaml deleted file mode 100644 index 311f9726a..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/mutating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: MutatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.mutatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/podmonitor.yaml deleted file mode 100644 index 1fe34fcb3..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/podmonitor.yaml +++ /dev/null @@ -1,27 +0,0 @@ -{{- if .Values.engine.sidecar.podMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: PodMonitor -metadata: - name: {{ include "airlock-microgateway.fullname" . }}-engine - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.engine.sidecar.podMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - namespaceSelector: - any: true - selector: - matchLabels: - sidecar.microgateway.airlock.com/inject: "true" - microgateway.airlock.com/managedBy: {{ .Release.Namespace }} - podMetricsEndpoints: - - targetPort: 19002 - path: /metrics - scheme: http -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/role.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/role.yaml deleted file mode 100644 index 5378be8ef..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/role.yaml +++ /dev/null @@ -1,45 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: Role -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -rules: - - apiGroups: - - "" - resources: - - configmaps - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - coordination.k8s.io - resources: - - leases - verbs: - - get - - list - - watch - - create - - update - - patch - - delete - - apiGroups: - - "" - resources: - - events - verbs: - - create - - patch -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/rolebinding.yaml deleted file mode 100644 index bafec1015..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/rolebinding.yaml +++ /dev/null @@ -1,20 +0,0 @@ -{{- if .Values.operator.rbac.create -}} -apiVersion: rbac.authorization.k8s.io/v1 -kind: RoleBinding -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: Role - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election -subjects: - - kind: ServiceAccount - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/selfsigned-issuer.yaml deleted file mode 100644 index 466c56338..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/selfsigned-issuer.yaml +++ /dev/null @@ -1,13 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Issuer -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selfSigned: {} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/serviceaccount.yaml deleted file mode 100644 index 434d7e9d3..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/serviceaccount.yaml +++ /dev/null @@ -1,13 +0,0 @@ -{{- if .Values.operator.serviceAccount.create -}} -apiVersion: v1 -kind: ServiceAccount -metadata: - name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with mustMerge .Values.operator.serviceAccount.annotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/servicemonitor.yaml deleted file mode 100644 index ff85a9a31..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/servicemonitor.yaml +++ /dev/null @@ -1,60 +0,0 @@ -{{- if .Values.operator.serviceMonitor.create }} -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - matchExpressions: - - { key: "operator.microgateway.airlock.com/isLeader", operator: DoesNotExist } - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: drop ---- -apiVersion: monitoring.coreos.com/v1 -kind: ServiceMonitor -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-leader - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceMonitor.labels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - selector: - matchLabels: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} - operator.microgateway.airlock.com/isLeader: "true" - endpoints: - - path: /metrics - port: metrics - scheme: http - metricRelabelings: - - sourceLabels: - - __name__ - regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} - action: keep -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/serving-certificate.yaml deleted file mode 100644 index 60b92e1e2..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/serving-certificate.yaml +++ /dev/null @@ -1,19 +0,0 @@ -apiVersion: cert-manager.io/v1 -kind: Certificate -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - dnsNames: - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc - - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc.cluster.local - issuerRef: - kind: Issuer - name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer - secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/validating-webhook.yaml deleted file mode 100644 index 5d6b4396b..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/validating-webhook.yaml +++ /dev/null @@ -1,28 +0,0 @@ -apiVersion: admissionregistration.k8s.io/v1 -kind: ValidatingWebhookConfiguration -metadata: - name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - annotations: - cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert - {{- with .Values.commonAnnotations }} - {{- toYaml . | nindent 4 }} - {{- end }} -webhooks: -{{- range $webhook := (include "airlock-microgateway-operator.validatingWebhooks" .) | fromYamlArray }} -- {{ toYaml $webhook | indent 2 | trim }} - {{- with $.Values.operator.watchNamespaceSelector }} - namespaceSelector: - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with $.Values.operator.watchNamespaces }} - namespaceSelector: - matchExpressions: - - key: kubernetes.io/metadata.name - operator: In - values: - {{- toYaml . | nindent 10 }} - {{- end }} -{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/webhook-service.yaml deleted file mode 100644 index 477ea839f..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/webhook-service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-webhook - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: https - name: webhook - port: 443 - protocol: TCP - targetPort: 9443 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.3.4/templates/operator/xds-service.yaml deleted file mode 100644 index 81b41acf5..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/operator/xds-service.yaml +++ /dev/null @@ -1,24 +0,0 @@ -apiVersion: v1 -kind: Service -metadata: - name: airlock-microgateway-operator-xds - namespace: {{ .Release.Namespace }} - labels: - {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} - {{- with .Values.operator.serviceLabels }} - {{- toYaml . | nindent 4 }} - {{- end }} - {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} - annotations: - {{- toYaml . | nindent 4 }} - {{- end }} -spec: - ports: - - appProtocol: grpc - name: xds - port: 13377 - protocol: TCP - targetPort: 13377 - selector: - {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} - operator.microgateway.airlock.com/isLeader: "true" diff --git a/charts/airlock/microgateway/4.3.4/templates/scc-role.yaml b/charts/airlock/microgateway/4.3.4/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/scc-rolebinding.yaml b/charts/airlock/microgateway/4.3.4/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/serviceaccount.yaml b/charts/airlock/microgateway/4.3.4/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway/4.3.4/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.3.4/templates/tests/rbac.yaml index 93bd4cd1b..744799333 100644 --- a/charts/airlock/microgateway/4.3.4/templates/tests/rbac.yaml +++ b/charts/airlock/microgateway/4.3.4/templates/tests/rbac.yaml @@ -2,142 +2,63 @@ apiVersion: v1 kind: ServiceAccount metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests --- apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests roleRef: apiGroup: rbac.authorization.k8s.io kind: Role - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" subjects: - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" --- apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests rules: - apiGroups: - - microgateway.airlock.com + - "apps" resources: - - sidecargateways + - daemonsets resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + - {{ include "airlock-microgateway-cni.fullname" . }} verbs: - - get - - list - - watch - - delete + - get + - watch + - list - apiGroups: - - microgateway.airlock.com + - "" resources: - - sidecargateways + - pods + - pods/log verbs: - - create + - get + - list +{{- if .Values.rbac.createSCCRole }} - apiGroups: - - "" - resources: - - events - verbs: - - list -- apiGroups: - - "apps" - resources: - - deployments + - security.openshift.io resourceNames: - - "{{ include "airlock-microgateway.operator.fullname" . }}" - verbs: - - get - - list - - watch -- apiGroups: - - "apps" + - privileged resources: - - statefulsets - - statefulsets/scale - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + - securitycontextconstraints verbs: - - get - - list - - watch - - patch -- apiGroups: - - "" - resources: - - pods - - pods/log - - pods/status - - pods/attach - resourceNames: - - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" - - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" - - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" - verbs: - - get - - list - - create - - watch - - delete -- apiGroups: - - "" - resources: - - pods - verbs: - - create -{{- if .Values.operator.watchNamespaceSelector }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRoleBinding -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -roleRef: - apiGroup: rbac.authorization.k8s.io - kind: ClusterRole - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -subjects: - - kind: ServiceAccount - name: "{{ include "airlock-microgateway.fullname" . }}-tests" - namespace: {{ .Release.Namespace }} ---- -apiVersion: rbac.authorization.k8s.io/v1 -kind: ClusterRole -metadata: - labels: - app.kubernetes.io/component: tests - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" -rules: -- apiGroups: - - "" - resources: - - namespaces - verbs: - - get - - list -{{- end }} + - use +{{- end -}} {{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/templates/tests/service.yaml b/charts/airlock/microgateway/4.3.4/templates/tests/service.yaml deleted file mode 100644 index 30ddc278d..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/tests/service.yaml +++ /dev/null @@ -1,23 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: v1 -kind: Service -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-service" - namespace: {{ .Release.Namespace }} - labels: - app: test-service - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - selector: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} - ports: - - name: http - port: 8080 - targetPort: 8080 -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.3.4/templates/tests/statefulset.yaml deleted file mode 100644 index 710a7b9f6..000000000 --- a/charts/airlock/microgateway/4.3.4/templates/tests/statefulset.yaml +++ /dev/null @@ -1,56 +0,0 @@ -{{- if .Values.tests.enabled -}} -apiVersion: apps/v1 -kind: StatefulSet -metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - namespace: {{ .Release.Namespace }} - labels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} -spec: - serviceName: nginx - replicas: 0 - selector: - matchLabels: - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 6 }} - template: - metadata: - annotations: - k8s.v1.cni.cncf.io/networks: default/airlock-microgateway-cni - labels: - sidecar.microgateway.airlock.com/inject: "true" - sidecar.istio.io/inject: "false" - app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" - {{- include "airlock-microgateway.sharedLabels" . | nindent 8 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 8 }} - spec: - containers: - - image: cgr.dev/chainguard/nginx - name: nginx - ports: - - containerPort: 8080 - volumeMounts: - - mountPath: /var/lib/nginx/tmp/ - name: nginx-tmp - - mountPath: /var/run - name: nginx-run - securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 12 }} - {{- with .Values.imagePullSecrets }} - imagePullSecrets: - {{- toYaml . | nindent 8 }} - {{- end }} - volumes: - - emptyDir: {} - name: nginx-tmp - - emptyDir: {} - name: nginx-run -{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.3.4/templates/tests/test-install.yaml index ab82abea7..12d8c8de7 100644 --- a/charts/airlock/microgateway/4.3.4/templates/tests/test-install.yaml +++ b/charts/airlock/microgateway/4.3.4/templates/tests/test-install.yaml @@ -2,14 +2,11 @@ apiVersion: v1 kind: Pod metadata: - name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" namespace: {{ .Release.Namespace }} labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} app.kubernetes.io/component: test-install - app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests - sidecar.istio.io/inject: "false" - {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} - {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} annotations: helm.sh/hook: test helm.sh/hook-delete-policy: before-hook-creation @@ -19,209 +16,88 @@ spec: - name: test image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" securityContext: - {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true command: - sh - -c - | set -eu - clean_up() { - echo "" - echo "### Clean up test resources" - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - echo "" - echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s - sleep 3s - echo "" - } - fail() { + echo "Error: ${1}" echo "" - echo "### Error: ${1}" - echo "" - - if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then - echo "" - echo 'Microgateway Sidecargateway status:' - kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true - echo "" - echo "" - fi - - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then - echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" - kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true - echo "" - echo "" - echo 'Logs of Nginx container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true - echo "" - echo "" - # Wait for engine logs - sleep 10s - echo 'Logs of Microgateway Engine container:' - kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true - fi - + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer exit 1 } - create_sidecargateway() { - # create SidecarGateway resource for testing purposes - kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true - kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done - kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request - } - - {{- if .Values.operator.watchNamespaceSelector }} - echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" - if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then - labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') - fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` - .Release.Namespace - (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) - }} + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' fi - echo "" - {{- end }} - trap clean_up EXIT - echo "" - - echo "### Waiting for Microgateway Operator Deployments to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ - deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then - fail 'Timout occurred' + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' fi - echo "" - echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" - # scale to zero replicas to ensure no pods are present from previous runs - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s - kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s - echo "" - - echo "### Waiting for backend pod" - i=0 - while true; do - if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then - break - elif [ $i -gt 3 ]; then - fail 'Pod not ready' - fi - sleep 2s - i=$((i+1)) - done - - echo "### Checking Microgateway Engine sidecar container was injected" - if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then - fail 'Microgateway Engine sidecar container not injected' + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' fi - echo "True" - echo "" - echo "### Checking for valid license" - i=0 - while true; do - if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then - break - elif [ $i -gt 30 ]; then - fail 'Microgateway license is missing or invalid' - fi - sleep 2s - i=$((i+1)) - done - echo "True" - echo "" + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac - echo "### Create SidecarGateway resource for testing" - if ! create_sidecargateway ; then - fail 'Creation of SidecarGateway resource failed' - fi - echo "" - - echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" - if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then - fail 'Timout occurred' - fi - echo "" - - echo "### Waiting for 'engine-config-valid' condition" - if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then - fail 'Configuration was never accepted by the Microgateway Engine' - fi - sleep 5s - echo "" - echo "" - - echo "### Checking whether a valid request is successful and returns HTTP status code '200'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "200 OK"; then - fail 'A valid request was not successful' - fi - echo "" - echo "" - - echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" - out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) - echo "Response:" - echo "${out}" - if ! echo "${out}" | grep -q "400 Bad Request"; then - fail 'A malicious request was not blocked' - fi - echo "" - echo "" - - echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" - exit 0 - serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir {{- end -}} diff --git a/charts/airlock/microgateway/4.3.4/values.schema.json b/charts/airlock/microgateway/4.3.4/values.schema.json index 173d6b084..e087bd700 100644 --- a/charts/airlock/microgateway/4.3.4/values.schema.json +++ b/charts/airlock/microgateway/4.3.4/values.schema.json @@ -14,15 +14,6 @@ "commonAnnotations": { "$ref": "#/definitions/StringMap" }, - "crds": { - "type": "object", - "properties": { - "skipVersionCheck": { - "type": "boolean" - } - }, - "additionalProperties": false - }, "imagePullSecrets": { "type": "array", "items": { @@ -39,304 +30,120 @@ "additionalProperties": true } }, - "operator": { + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { "type": "object", "properties": { - "replicaCount": { - "type": "integer", - "minimum": 0 - }, - "updateStrategy": { - "$ref": "#/definitions/UpdateStrategy" - }, - "image": { - "$ref": "#/definitions/Image" - }, - "podAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "podLabels": { - "$ref": "#/definitions/StringMap" - }, - "serviceAnnotations": { - "$ref": "#/definitions/StringMap" - }, - "serviceLabels": { - "$ref": "#/definitions/StringMap" - }, - "resources": { - "type": "object" - }, - "nodeSelector": { - "$ref": "#/definitions/StringMap" - }, - "tolerations": { - "type": "array", - "items": { - "type": "object" - } - }, - "affinity": { - "type": "object" - }, - "config": { - "type": "object", - "properties": { - "logLevel": { - "type": "string", - "enum": [ - "debug", - "info", - "warn", - "error" - ] - } - }, - "required": [ - "logLevel" - ], - "additionalProperties": false - }, - "serviceAccount": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "annotations": { - "$ref": "#/definitions/StringMap" - }, - "name": { - "type": "string" - } - }, - "required": [ - "annotations", - "create", - "name" - ], - "additionalProperties": false - }, - "watchNamespaces": { - "type": "array", - "items": { - "type": "string" - } - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - }, - "rbac": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - }, - "serviceMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "oneOf": [ - { - "properties": { - "watchNamespaces": { - "minItems": 1 - }, - "watchNamespaceSelector": { - "additionalProperties": false - } - } - }, - { - "properties": { - "watchNamespaces": { - "maxItems": 0 - }, - "watchNamespaceSelector": { - "$ref": "#/definitions/LabelSelector" - } - } - } - ], - "required": [ - "affinity", - "config", - "image", - "updateStrategy", - "nodeSelector", - "podAnnotations", - "podLabels", - "rbac", - "replicaCount", - "resources", - "serviceAccount", - "serviceAnnotations", - "serviceLabels", - "serviceMonitor", - "tolerations" - ], - "additionalProperties": false - }, - "engine": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - }, - "sidecar": { - "type": "object", - "properties":{ - "podMonitor": { - "type": "object", - "properties": { - "create": { - "type": "boolean" - }, - "labels": { - "$ref": "#/definitions/StringMap" - } - }, - "required": [ - "create" - ], - "additionalProperties": false - } - }, - "required": [ - "podMonitor" - ], - "additionalProperties": false - } - }, - "required": [ - "image", - "resources", - "sidecar" - ], - "additionalProperties": false - }, - "networkValidator": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - } - }, - "required": [ - "image" - ], - "additionalProperties": false - }, - "sessionAgent": { - "type": "object", - "properties": { - "image": { - "$ref": "#/definitions/Image" - }, - "resources": { - "type": "object" - } - }, - "required": [ - "image", - "resources" - ], - "additionalProperties": false - }, - "license": { - "type": "object", - "properties": { - "secretName": { - "type": "string", - "minLength": 1 - } - }, - "required": [ - "secretName" - ], - "additionalProperties": false - }, - "dashboards": { - "type": "object", - "properties" : { "create": { "type": "boolean" }, - "config": { - "type": "object", - "properties": { - "grafana": { - "type": "object", - "properties": { - "folderAnnotation": { - "$ref": "#/definitions/NameValuePair" - }, - "dashboardLabel": { - "$ref": "#/definitions/NameValuePair" - } - }, - "required": [ - "folderAnnotation", - "dashboardLabel" - ], - "additionalProperties": false - } - }, - "required": [ - "grafana" - ], - "additionalProperties": false - }, - "instances": { - "type": "object", - "properties": { - "overview": { - "$ref": "#/definitions/DashboardInstance" - }, - "license" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockMetrics" : { - "$ref": "#/definitions/DashboardInstance" - }, - "blockLogs" : { - "$ref": "#/definitions/DashboardInstance" - } - }, - "required": [ - "overview", - "license", - "blockMetrics", - "blockLogs" - ], - "additionalProperties": false + "createSCCRole": { + "type": "boolean" } }, "required": [ "create", - "config", - "instances" + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" ], "additionalProperties": false }, @@ -357,18 +164,22 @@ } }, "required": [ + "affinity", "commonAnnotations", "commonLabels", - "crds", - "engine", + "config", "fullnameOverride", + "image", "imagePullSecrets", - "license", + "multusNetworkAttachmentDefinition", "nameOverride", - "operator", - "networkValidator", - "sessionAgent", - "dashboards", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", "tests" ], "additionalProperties": false, @@ -409,132 +220,6 @@ "tag" ], "additionalProperties": false - }, - "LabelSelector": { - "type": "object", - "properties": { - "matchExpressions": { - "type": "array", - "items": { - "type": "object", - "required": [ - "key", - "operator" - ], - "properties": { - "key": { - "type": "string" - }, - "operator": { - "type": "string" - }, - "values": { - "type": "array", - "items": { - "type": "string" - } - } - }, - "additionalProperties": false - } - }, - "matchLabels": { - "$ref": "#/definitions/StringMap" - } - }, - "additionalProperties": false - }, - "UpdateStrategy": { - "type": "object", - "oneOf" : [ - { - "properties": { - "type": { - "$ref": "#/definitions/RecreateType" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - }, - { - "properties": { - "type": { - "$ref": "#/definitions/RollingUpdateType" - }, - "rollingUpdate": { - "$ref": "#/definitions/RollingUpdate" - } - }, - "required": [ - "type" - ], - "additionalProperties": false - } - ] - }, - "RecreateType": { - "type": "string", - "enum": [ - "Recreate" - ] - }, - "RollingUpdateType": { - "type": "string", - "enum": [ - "RollingUpdate" - ] - }, - "RollingUpdate": { - "type": "object", - "properties": { - "maxSurge": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - }, - "maxUnavailable": { - "type": ["integer", "string"], - "minimum": 0, - "pattern": "^\\d+%?$" - } - }, - "anyOf": [ - {"required": ["maxSurge"]}, - {"required": ["maxUnavailable"]} - ], - "additionalProperties": false - }, - "DashboardInstance" : { - "type" : "object", - "properties" : { - "create" : { - "type" : "boolean" - } - }, - "required" : [ - "create" - ], - "additionalProperties": false - }, - "NameValuePair" : { - "type" : "object", - "properties" : { - "name" : { - "type": "string", - "minLength": 1 - }, - "value" : { - "type" : "string", - "minLength": 1 - } - }, - "required" : [ - "name", - "value" - ], - "additionalProperties": false } } } diff --git a/charts/airlock/microgateway/4.3.4/values.yaml b/charts/airlock/microgateway/4.3.4/values.yaml index af720d5c6..63ef36033 100644 --- a/charts/airlock/microgateway/4.3.4/values.yaml +++ b/charts/airlock/microgateway/4.3.4/values.yaml @@ -1,4 +1,4 @@ -# -- Allows overriding the name to use instead of "microgateway". +# -- Allows overriding the name to use instead of "microgateway-cni". nameOverride: "" # -- Allows overriding the name to use as full name of resources. fullnameOverride: "" @@ -10,203 +10,75 @@ commonAnnotations: {} imagePullSecrets: [] # - name: myRegistryKeySecretName -crds: - # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. - # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster - # when performing a "helm install/upgrade". - skipVersionCheck: false -operator: - # -- Number of replicas for the operator Deployment. - replicaCount: 2 - # -- Specifies the operator update strategy. - updateStrategy: - type: RollingUpdate - # Specifies the Airlock Microgateway Operator image. - image: - # -- Image repository from which to pull the Airlock Microgateway Operator image. - repository: "quay.io/airlock/microgateway-operator" - # -- Image tag to pull. - tag: "4.3.4" - # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). - # Overrides tag when specified. - digest: "sha256:6819c78d5570de66edce6c13964c6e1b4cc4746d0c0bc6f4975cd38e324828c0" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Annotations to add to all Pods. - podAnnotations: {} - # -- Labels to add to all Pods. - podLabels: {} - # -- Annotations to add to the Service. - serviceAnnotations: {} - # prometheus.io/scrape: "true" - # prometheus.io/port: "8080" - - # -- Labels to add to the Service. - serviceLabels: {} - # -- Resource restrictions to apply to the operator container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 1000m - # memory: 512Mi - # requests: - # cpu: 100m - # memory: 512Mi - - # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. - nodeSelector: {} - # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. - tolerations: [] - # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. - affinity: {} - # Parameters for the operator configuration. - config: - # -- Operator application log level. - logLevel: "info" - # Configures the generation of the ServiceAccount. - serviceAccount: - # -- Whether a ServiceAccount should be created. - create: true - # -- Annotations to add to the ServiceAccount. - annotations: {} - # -- Name of the ServiceAccount to use. - # If not set and create is true, a name is generated using the fullname template. - name: "" - # -- Allows to restrict the operator to specific namespaces, depending on your needs. - # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). - # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. - # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. - # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. - # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. - # Please note that this feature requires a Premium license. - watchNamespaces: [] - # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. - # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. - # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). - # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. - # Please note that this feature requires a Premium license. - watchNamespaceSelector: {} - # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. - # matchLabels: - # microgateway.airlock.com/enable: "true" - # matchExpressions: - # - { key: environment, operator: NotIn, values: [dev] } - - # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. - rbac: - # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. - create: true - # Configures the generation of a Prometheus Operator ServiceMonitor. - serviceMonitor: - # -- Whether to create a ServiceMonitor resource for monitoring. - create: false - # -- Labels to add to the ServiceMonitor. - labels: {} - # release: "" -engine: - # Specifies the Airlock Microgateway Engine image. - image: - # -- Image repository from which to pull the Airlock Microgateway Engine image. - repository: "quay.io/airlock/microgateway-engine" - # -- Image tag to pull. - tag: "4.3.4" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:91e05c509bed3b51ff4888d7475980d56cbc85db121aa766d1bde413204f9070" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Engine container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 500m - # memory: 128Mi - # requests: - # cpu: 10m - # memory: 40Mi - - # Additional configuration when deployed as a sidecar. - sidecar: - # Configures the generation of a Prometheus Operator PodMonitor. - podMonitor: - # -- Whether to create a PodMonitor resource for monitoring. - create: false - # -- Labels to add to the PodMonitor. - labels: {} - # release: "" -networkValidator: - # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. - image: - # -- Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. - repository: "cgr.dev/chainguard/netcat" - # -- Image tag to pull. - tag: "" - # -- SHA256 image digest to pull (in the format "sha256:7a73d4b82a2d4165bbc5efa55de4fee9d43f2b1c1edb3505cdc8afd1361bad9b"). - # Overrides tag when specified. - digest: "sha256:7a73d4b82a2d4165bbc5efa55de4fee9d43f2b1c1edb3505cdc8afd1361bad9b" - # -- Pull policy for this image. - pullPolicy: IfNotPresent -sessionAgent: - # Specifies the Airlock Microgateway Session Agent image. - image: - # -- Image repository from which to pull the Airlock Microgateway Session Agent image. - repository: "quay.io/airlock/microgateway-session-agent" - # -- Image tag to pull. - tag: "4.3.4" - # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). - # Overrides tag when specified. - digest: "sha256:df4e50d0929cb4c5e4486452979b59ec17f5e49a1516b685acd3a1ab0ddb3cf4" - # -- Pull policy for this image. - pullPolicy: IfNotPresent - # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. - resources: {} - # We recommend at least the following resource specification. - # limits: - # cpu: 150m - # memory: 32Mi - # requests: - # cpu: 10m - # memory: 8Mi -license: - # -- Name of the secret containing the "microgateway-license.txt" key. - secretName: "airlock-microgateway-license" -# Creates dashboards in the form of ConfigMaps that can be imported -# by Grafana using its sidecar setup. -dashboards: - # -- Whether to create any ConfigMaps containing Grafana dashboards to import. +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.3.4" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:1e01310b3ad8566e9b39ee539ed5c959049aadda1a18c1a534e96d8865e20172" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. create: false - config: - # Configures the necessary label and annotations along with their values - # to enable Grafana to correctly identify the ConfigMaps containing - # dashboards and file them within a dedicated folder in the dashboard overview. - # These settings need to match the Grafana sidecar configuration. - grafana: - folderAnnotation: - # -- Name of the annotation containing the folder name to file dashboards into. - name: "grafana_folder" - # -- Name of the folder dashboards are filed into within the Grafana UI. - value: "Airlock Microgateway" - dashboardLabel: - # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. - name: "grafana_dashboard" - # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. - value: "1" - instances: - # Available dashboard instances that can be individually created/deployed. - overview: - # -- Whether to create the overview dashboard. - create: true - license: - # -- Whether to create the license dashboard. - create: true - blockMetrics: - # -- Whether to create the block metrics dashboard. - create: true - blockLogs: - # -- Whether to create the block logs dashboard. - create: true -# Check whether the installation of the Airlock Microgateway Helm Chart was successful. -# Requires a secret with a valid Airlock Microgateway license key already to be present. + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system tests: # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). # If set to false, `helm test` will not run any tests. diff --git a/charts/airlock/microgateway/4.4.1/.helmignore b/charts/airlock/microgateway/4.4.1/.helmignore new file mode 100644 index 000000000..101ff5ac5 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/.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 +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ +# CRDs kustomization.yaml +/crds/kustomization.yaml +# Helm unit tests +/tests +/validation diff --git a/charts/airlock/microgateway/4.4.1/Chart.yaml b/charts/airlock/microgateway/4.4.1/Chart.yaml new file mode 100644 index 000000000..4e46fe73f --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/Chart.yaml @@ -0,0 +1,44 @@ +annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.4/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: "" + charts.openshift.io/name: Airlock Microgateway +apiVersion: v2 +appVersion: 4.4.1 +description: A Helm chart for deploying the Airlock Microgateway +home: https://www.airlock.com/en/microgateway +icon: file://assets/icons/microgateway.svg +keywords: +- WAF +- Web Application Firewall +- WAAP +- Web Application and API protection +- OWASP +- Airlock +- Microgateway +- Security +- Filtering +- DevSecOps +- shift left +- control plane +- Operator +kubeVersion: '>=1.25.0-0' +maintainers: +- email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ +name: microgateway +sources: +- https://github.com/airlock/microgateway +type: application +version: 4.4.1 diff --git a/charts/airlock/microgateway/4.4.1/README.md b/charts/airlock/microgateway/4.4.1/README.md new file mode 100644 index 000000000..1c976c66c --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/README.md @@ -0,0 +1,186 @@ +# Airlock Microgateway + +![Version: 4.4.1](https://img.shields.io/badge/Version-4.4.1-informational?style=flat-square) ![AppVersion: 4.4.1](https://img.shields.io/badge/AppVersion-4.4.1-informational?style=flat-square) + +*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* + + + + + Microgateway + + +Modern application security is embedded in the development workflow and follows DevSecOps paradigms. Airlock Microgateway is the perfect fit for these requirements. It is a lightweight alternative to the Airlock Gateway appliance, optimized for Kubernetes environments. Airlock Microgateway protects your applications and microservices with the tried-and-tested Airlock security features against attacks, while also providing a high degree of scalability. +__This Helm chart is part of Airlock Microgateway. See our [GitHub repo](https://github.com/airlock/microgateway/tree/4.4.1).__ + +### Features +* Kubernetes native integration with sidecar injection and Gateway API support +* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction +* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication +* Content security filters for protecting against known attacks (OWASP Top 10) +* Access control using OpenID Connect to allow only authenticated users to access the protected services +* API security features like JSON parsing, OpenAPI specification enforcement or GraphQL schema validation + +For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. + +## Documentation and links + +Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. + +* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) +* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) +* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) +* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) +* [GitHub](https://github.com/airlock/microgateway) + +# Quick start guide + +The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. + +## Prerequisites +* (Recommended) [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Required for [data plane mode sidecar](https://docs.airlock.com/microgateway/latest/?topic=MGW-00000137)) +* [Airlock Microgateway License](#obtain-airlock-microgateway-license) +* [cert-manager](https://cert-manager.io/) +* [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) + +In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. +For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. +### Obtain Airlock Microgateway License +1. Either request a community or premium license + * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) + * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) +2. Check your inbox and save the license file microgateway-license.txt locally. + +> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. +### Deploy cert-manager +```bash +helm repo add jetstack https://charts.jetstack.io +helm install cert-manager jetstack/cert-manager --version 'v1.16.1' -n cert-manager --create-namespace --set crds.enabled=true --wait +``` + +## Deploy Airlock Microgateway Operator + +> This guide assumes a microgateway-license.txt file is present in the working directory. + +1. Install CRDs and Operator. + ```bash + # Create namespace + kubectl create namespace airlock-microgateway-system + + # Install License + kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt + + # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) + helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.4.1' --wait + ``` + +2. (Recommended) You can verify the correctness of the installation with `helm test`. + ```bash + helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.4.1' + helm test airlock-microgateway -n airlock-microgateway-system --logs + helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.4.1' + ``` + +### Upgrading CRDs + +The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. +CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: +```bash +kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.4.1 --server-side --force-conflicts +``` + +**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + +## Support + +### Premium support +If you have a paid license, please follow the [premium support process](https://techzone.ergon.ch/support-process). + +### Community support +For the community edition, check our **[Airlock community forum](https://forum.airlock.com/)** for FAQs or register to post your question. +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| commonAnnotations | object | `{}` | Annotations to add to all resources. | +| commonLabels | object | `{}` | Labels to add to all resources. | +| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | +| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | +| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | +| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | +| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | +| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | +| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | +| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | +| dashboards.instances.headerLogs.create | bool | `true` | Whether to create the header rewrite logs dashboard. | +| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | +| dashboards.instances.logOnlyLogs.create | bool | `true` | Whether to create the log only logs dashboard. | +| dashboards.instances.logOnlyMetrics.create | bool | `true` | Whether to create the log only metrics dashboard | +| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | +| engine.image.digest | string | `"sha256:06573ef5e6769dbd6eb8606e34c56f1ad2084b6adcae9925b1d2d153a45cbc47"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | +| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | +| engine.image.tag | string | `"4.4.1"` | Image tag to pull. | +| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | +| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | +| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | +| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | +| networkValidator.image.digest | string | `"sha256:7ef657ce316ce9d86f90c1dc99702d1190877c6ac2e923e696dc82c30050a14c"` | SHA256 image digest to pull (in the format "sha256:7ef657ce316ce9d86f90c1dc99702d1190877c6ac2e923e696dc82c30050a14c"). Overrides tag when specified. | +| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| networkValidator.image.repository | string | `"cgr.dev/chainguard/netcat"` | Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. | +| networkValidator.image.tag | string | `""` | Image tag to pull. | +| networkValidator.resources | object | `{"limits":{"cpu":"25m","memory":"12Mi"},"requests":{"cpu":"5m","memory":"1Mi"}}` | Resource restrictions to apply to the Airlock Microgateway Network Validator init-container. | +| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | +| operator.config.logLevel | string | `"info"` | Operator application log level. | +| operator.gatewayAPI.controllerName | string | `"microgateway.airlock.com/gatewayclass-controller"` | Controller name referred in the GatewayClasses managed by this operator. The value must be a path prefixed by the domain `microgateway.airlock.com`. | +| operator.gatewayAPI.enabled | bool | `false` | Whether to enable the Kubernetes Gateway API related controllers. Requires that the gateway.networking.k8s.io/v1 resources are installed on the cluster. | +| operator.image.digest | string | `"sha256:1133c3e59418eec1721683e68dd19faca577609ace6eebd010a56e52b1f75789"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | +| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | +| operator.image.tag | string | `"4.4.1"` | Image tag to pull. | +| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | +| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | +| operator.podLabels | object | `{}` | Labels to add to all Pods. | +| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | +| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | +| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | +| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | +| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | +| operator.serviceLabels | object | `{}` | Labels to add to the Service. | +| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | +| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | +| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | +| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | +| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | +| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | +| sessionAgent.image.digest | string | `"sha256:733a25f61ea7cf43c0a46da7d3ecb9a263bda49bf60e1fd8e4162be33aa24b7b"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | +| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | +| sessionAgent.image.tag | string | `"4.4.1"` | Image tag to pull. | +| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | + +## License +View the [detailed license terms](https://www.airlock.com/en/airlock-license) for the software contained in this image. +* Decompiling or reverse engineering is not permitted. +* Using any of the deny rules or parts of these filter patterns outside of the image is not permitted. + +Airlock® is a security innovation by [ergon](https://www.ergon.ch/en) + + + + + + + Airlock Secure Access Hub + + diff --git a/charts/airlock/microgateway/4.2.3/app-readme.md b/charts/airlock/microgateway/4.4.1/app-readme.md similarity index 100% rename from charts/airlock/microgateway/4.2.3/app-readme.md rename to charts/airlock/microgateway/4.4.1/app-readme.md diff --git a/charts/airlock/microgateway/4.4.1/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/accesscontrols.microgateway.airlock.com.yaml new file mode 100644 index 000000000..c10c65c1f --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/crds/accesscontrols.microgateway.airlock.com.yaml @@ -0,0 +1,501 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.4 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.4.1 + name: accesscontrols.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: AccessControl + listKind: AccessControlList + plural: accesscontrols + singular: accesscontrol + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: AccessControl specifies the options to perform access control with a Microgateway Engine container. + 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: Specifies how the Airlock Microgateway Engine performs access control. + properties: + policies: + description: Policies configures access control policies. The first matching policy (from top to bottom) applies. + items: + properties: + authorization: + description: Authorization configures how requests are authorized. An empty object value {} disables authorization. + properties: + authentication: + description: Authentication specifies that clients need to be authenticated with the provided method. + properties: + oidc: + description: OIDC configures client authentication using OpenID Connect. + properties: + oidcRelyingPartyRef: + description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - oidcRelyingPartyRef + type: object + type: object + deny: + description: Deny specifies to deny access for all requests matching this policy. + type: object + requireAll: + description: RequireAll specifies conditions which must all be satisfied for the request to be authorized. + items: + properties: + oidc: + description: OIDC specifies a condition on the result of an OpenID Connect flow. + properties: + claim: + description: Claim specifies a condition on a JWT claim. + properties: + name: + description: Name of the claim. + minLength: 1 + type: string + value: + description: |- + Value of the claim. If not specified, only existence of the claim is checked (any value is allowed). + + Value matching is only supported if the data type of the claim is either primitive (`number`, `boolean`, `string`) or `array` of primitives. + In case of a non-string value, the match will be performed against the stringified value. + + If the claim has an unsupported data type (e.g. `object` or `null`), its value will never match. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + required: + - name + type: object + required: + - claim + type: object + required: + - oidc + type: object + minItems: 1 + type: array + requireAny: + description: RequireAny specifies conditions of which at least one must be satisfied for the request to be authorized. + items: + properties: + oidc: + description: OIDC specifies a condition on the result of an OpenID Connect flow. + properties: + claim: + description: Claim specifies a condition on a JWT claim. + properties: + name: + description: Name of the claim. + minLength: 1 + type: string + value: + description: |- + Value of the claim. If not specified, only existence of the claim is checked (any value is allowed). + + Value matching is only supported if the data type of the claim is either primitive (`number`, `boolean`, `string`) or `array` of primitives. + In case of a non-string value, the match will be performed against the stringified value. + + If the claim has an unsupported data type (e.g. `object` or `null`), its value will never match. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + required: + - name + type: object + required: + - claim + type: object + required: + - oidc + type: object + minItems: 1 + type: array + type: object + identityPropagation: + description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. + properties: + actions: + description: Actions specifies the propagation actions. + items: + properties: + identityPropagationRef: + description: IdentityPropagationRef selects an IdentityPropagation to apply. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - identityPropagationRef + type: object + type: array + onFailure: + description: |- + OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: + _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. + enum: + - Pass + type: string + required: + - actions + - onFailure + type: object + requestConditions: + description: |- + RequestConditions defines additional request properties which must be matched in order for this policy to apply. A policy without request conditions will always match. + + WARNING: There is currently a limitation that if `authentication.oidc` is configured for this policy, you must ensure that the request condition also matches logout requests and callback redirects from the OIDC Provider as configured in the OIDCRelyingParty (`pathMapping.logoutPath` / `pathMapping.redirectPath`). + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - authorization + type: object + minItems: 1 + type: array + required: + - policies + type: object + required: + - spec + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/contentsecurities.microgateway.airlock.com.yaml similarity index 98% rename from charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/contentsecurities.microgateway.airlock.com.yaml index 05214f023..cbe6fb3a1 100644 --- a/charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/contentsecurities.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: contentsecurities.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.4.1/crds/contentsecuritypolicies.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/contentsecuritypolicies.microgateway.airlock.com.yaml new file mode 100644 index 000000000..3fd1d7545 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/crds/contentsecuritypolicies.microgateway.airlock.com.yaml @@ -0,0 +1,476 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.4 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.4.1 + gateway.networking.k8s.io/policy: direct + name: contentsecuritypolicies.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: ContentSecurityPolicy + listKind: ContentSecurityPolicyList + plural: contentsecuritypolicies + singular: contentsecuritypolicy + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: ContentSecurityPolicy is a Direct Attached Policy for the Kubernetes Gateway API. It specifies the options to secure an upstream web application with a Microgateway. + 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 ContentSecurityPolicy. + properties: + secured: + description: Secured enables WAF processing for the routes attached to this policy. + properties: + apiProtection: + description: |- + APIProtection defines the relevant configurations to protect APIs. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + graphQLRef: + description: |- + GraphQLRef selects the relevant GraphQL configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + openAPIRef: + description: |- + OpenAPIRef selects the relevant OpenAPI configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + filter: + description: |- + Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests + to protect against various attack patterns. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + denyRulesRef: + description: |- + DenyRulesRef selects the relevant DenyRules configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + limitsRef: + description: |- + LimitsRef selects the relevant Limits configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + parserRef: + description: |- + ParserRef selects the relevant Parser configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + targetRefs: + description: |- + TargetRefs are the resources this policy is being attached to. Referenced resources must be in the same namespace as the policy. + Support: HTTPRoute. + items: + description: |- + LocalPolicyTargetReference identifies an API object to apply a direct or + inherited policy to. This should be used as part of Policy resources + that can target Gateway API resources. For more information on how this + policy attachment model works, and a sample Policy resource, refer to + the policy attachment documentation for Gateway API. + properties: + group: + description: Group is the group of the target resource. + 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 target resource. + 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 target resource. + maxLength: 253 + minLength: 1 + type: string + required: + - group + - kind + - name + type: object + maxItems: 16 + minItems: 1 + type: array + x-kubernetes-validations: + - message: 'TargetRef Kind must be: HTTPRoute' + rule: self.all(t, t.kind=='HTTPRoute') + - message: TargetRef Group must be gateway.networking.k8s.io. + rule: self.all(t, t.group=='gateway.networking.k8s.io') + unsecured: + description: |- + Unsecured disables all WAF functionality and therefore protection for the routes attached to this policy. + WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. + type: object + required: + - targetRefs + type: object + status: + description: Status defines the state of the ContentSecurityPolicy. + properties: + ancestors: + description: |- + Ancestors is a list of ancestor resources (usually Gateways) that are + associated with the policy, and the status of the policy with respect to + each ancestor. When this policy attaches to a parent, the controller that + manages the parent and the ancestors MUST add an entry to this list when + the controller first sees the policy and SHOULD update the entry as + appropriate when the relevant ancestor is modified. + + Note that choosing the relevant ancestor is left to the Policy designers; + an important part of Policy design is designing the right object level at + which to namespace this status. + + Note also that implementations MUST ONLY populate ancestor status for + the Ancestor resources they are responsible for. Implementations MUST + use the ControllerName field to uniquely identify the entries in this list + that they are responsible for. + + Note that to achieve this, the list of PolicyAncestorStatus structs + MUST be treated as a map with a composite key, made up of the AncestorRef + and ControllerName fields combined. + + A maximum of 16 ancestors will be represented in this list. An empty list + means the Policy is not relevant for any ancestors. + + If this slice is full, implementations MUST NOT add further entries. + Instead they MUST consider the policy unimplementable and signal that + on any related resources such as the ancestor that would be referenced + here. For example, if this list was full on BackendTLSPolicy, no + additional Gateways would be able to reference the Service targeted by + the BackendTLSPolicy. + items: + description: |- + PolicyAncestorStatus describes the status of a route with respect to an + associated Ancestor. + + Ancestors refer to objects that are either the Target of a policy or above it + in terms of object hierarchy. For example, if a policy targets a Service, the + Policy's Ancestors are, in order, the Service, the HTTPRoute, the Gateway, and + the GatewayClass. Almost always, in this hierarchy, the Gateway will be the most + useful object to place Policy status on, so we recommend that implementations + SHOULD use Gateway as the PolicyAncestorStatus object unless the designers + have a _very_ good reason otherwise. + + In the context of policy attachment, the Ancestor is used to distinguish which + resource results in a distinct application of this policy. For example, if a policy + targets a Service, it may have a distinct result per attached Gateway. + + Policies targeting the same resource may have different effects depending on the + ancestors of those resources. For example, different Gateways targeting the same + Service may have different capabilities, especially if they have different underlying + implementations. + + For example, in BackendTLSPolicy, the Policy attaches to a Service that is + used as a backend in a HTTPRoute that is itself attached to a Gateway. + In this case, the relevant object for status is the Gateway, and that is the + ancestor object referred to in this status. + + Note that a parent is also an ancestor, so for objects where the parent is the + relevant object for status, this struct SHOULD still be used. + + This struct is intended to be used in a slice that's effectively a map, + with a composite key made up of the AncestorRef and the ControllerName. + properties: + ancestorRef: + description: |- + AncestorRef corresponds with a ParentRef in the spec that this + PolicyAncestorStatus 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). + + 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. + + There are two kinds of parent resources with "Core" support: + + * Gateway (Gateway conformance profile) + * Service (Mesh conformance profile, ClusterIP Services only) + + Support for other resources is Implementation-Specific. + 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. + + 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. + + 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. + + + ParentRefs from a Route to a Service in the same namespace are "producer" + routes, which apply default routing rules to inbound connections from + any namespace to the Service. + + ParentRefs from a Route to a Service in a different namespace are + "consumer" routes, and these routing rules are only applied to outbound + connections originating from the same namespace as the Route, for which + the intended destination of the connections are a Service targeted as a + ParentRef of the Route. + + + 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. + + 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. + + + When the parent resource is a Service, this targets a specific port in the + Service spec. When both Port (experimental) and SectionName are specified, + the name and port of the selected port must match both specified values. + + + Implementations MAY choose to support other parent resources. + Implementations supporting other types of parent resources MUST clearly + document how/if Port is interpreted. + + 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. + + 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: + + * Gateway: Listener name. When both Port (experimental) and SectionName + are specified, the name and port of the selected listener must match + both specified values. + * Service: Port name. When both Port (experimental) and SectionName + are specified, the name and port of the selected listener must match + both specified values. + + Implementations MAY choose to support attaching Routes to other resources. + If that is the case, they MUST clearly document how SectionName is + interpreted. + + 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. + + 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 + conditions: + description: Conditions describes the status of the Policy with respect to the given Ancestor. + items: + description: Condition contains details for one aspect of the current state of this API Resource. + 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. + 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. + + Example: "example.net/gateway-controller". + + 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). + + 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 + required: + - ancestorRef + - controllerName + type: object + maxItems: 16 + type: array + required: + - ancestors + type: object + required: + - spec + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/denyrules.microgateway.airlock.com.yaml similarity index 99% rename from charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/denyrules.microgateway.airlock.com.yaml index 906353c0a..234190a28 100644 --- a/charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/denyrules.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: denyrules.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -864,7 +864,9 @@ spec: TEMPLATE | UNIXCMD | WINCMD | - XSS + XSS | + SSRF | + BOT enum: - ENCODING - EXPLOIT @@ -883,6 +885,8 @@ spec: - UNIXCMD - WINCMD - XSS + - SSRF + - BOT type: string minItems: 1 type: array @@ -917,7 +921,9 @@ spec: TEMPLATE | UNIXCMD | WINCMD | - XSS + XSS | + SSRF | + BOT enum: - ENCODING - EXPLOIT @@ -936,6 +942,8 @@ spec: - UNIXCMD - WINCMD - XSS + - SSRF + - BOT type: string minItems: 1 type: array diff --git a/charts/airlock/microgateway/4.3.1/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/envoyclusters.microgateway.airlock.com.yaml similarity index 96% rename from charts/airlock/microgateway/4.3.1/crds/envoyclusters.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/envoyclusters.microgateway.airlock.com.yaml index 8d2c6ef97..4127d53eb 100644 --- a/charts/airlock/microgateway/4.3.1/crds/envoyclusters.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/envoyclusters.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 + app.kubernetes.io/version: 4.4.1 name: envoyclusters.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.1/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/envoyconfigurations.microgateway.airlock.com.yaml similarity index 98% rename from charts/airlock/microgateway/4.3.1/crds/envoyconfigurations.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/envoyconfigurations.microgateway.airlock.com.yaml index 38f381b72..a71ef4cc2 100644 --- a/charts/airlock/microgateway/4.3.1/crds/envoyconfigurations.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/envoyconfigurations.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 + app.kubernetes.io/version: 4.4.1 name: envoyconfigurations.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.2.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml similarity index 96% rename from charts/airlock/microgateway/4.2.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml index f81221802..358e1973d 100644 --- a/charts/airlock/microgateway/4.2.3/crds/envoyhttpfilters.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/envoyhttpfilters.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.14.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.2.3 + app.kubernetes.io/version: 4.4.1 name: envoyhttpfilters.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.3/crds/graphqls.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/graphqls.microgateway.airlock.com.yaml similarity index 97% rename from charts/airlock/microgateway/4.3.3/crds/graphqls.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/graphqls.microgateway.airlock.com.yaml index 1d9cb3b94..39046d24c 100644 --- a/charts/airlock/microgateway/4.3.3/crds/graphqls.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/graphqls.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 + app.kubernetes.io/version: 4.4.1 name: graphqls.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.4.1/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/headerrewrites.microgateway.airlock.com.yaml new file mode 100644 index 000000000..d99797f1e --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/crds/headerrewrites.microgateway.airlock.com.yaml @@ -0,0 +1,2083 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.4 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.4.1 + name: headerrewrites.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: HeaderRewrites + listKind: HeaderRewritesList + plural: headerrewrites + singular: headerrewrites + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: HeaderRewrites is the Schema for the headerrewrites 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: Specification of the desired header rewriting behavior. + properties: + request: + description: Request defines manipulations on upstream request headers. + properties: + add: + description: Add defines which request headers will be added before forwarding to the upstream. + properties: + custom: + description: |- + Custom allows configuring additional upstream request headers. + Add selected headers. + items: + properties: + headers: + description: Headers to add. + items: + description: HeaderRewritesHeader specifies a header with a particular value + properties: + name: + description: Name defines the name of a header. + minLength: 1 + type: string + value: + description: Value defines the value of a header. + type: string + required: + - name + - value + type: object + minItems: 1 + type: array + mode: + default: AddIfAbsent + description: Mode defines the header addition strategy. + enum: + - AddIfAbsent + - OverwriteOrAdd + type: string + name: + description: Name describing the configured operation. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + allow: + description: |- + Allow defines which request headers will be forwarded to the upstream. + This can either be allHeaders or matchingHeaders. + Default: matchingHeaders: {...} + properties: + allHeaders: + description: AllHeaders specifies that all request headers should be forwarded. + type: object + matchingHeaders: + description: MatchingHeaders specifies which request headers should be forwarded. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream request headers. + properties: + standardHeaders: + default: true + description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. + type: boolean + tracingHeaders: + default: false + description: TracingHeaders defines whether to allow common tracing headers to be forwarded to the upstream. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream request headers. + items: + properties: + headers: + description: Headers to allow. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + remove: + description: Remove defines which request headers will be removed before forwarding to the upstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream request headers. + properties: + alternativeForwardedHeaders: + default: true + description: |- + AlternativeForwardedHeaders removes downstream request headers which could potentially + be abused to alter the upstream's view of the remote connection. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream request headers. + items: + properties: + headers: + description: Headers to remove. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + response: + description: Response defines manipulations on upstream response headers. + properties: + add: + description: Add defines which response headers will be added before forwarding to the downstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response headers. + properties: + csp: + default: true + description: |- + CSP sets a content security policy which allows only same-origin requests except for images + if the 'Content-Security-Policy' header is not set by the upstream. + type: boolean + featurePolicy: + default: false + description: |- + FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features + if the 'Feature-Policy' header is not set by the upstream. + **Deprecated:** Use permissionsPolicy instead. + type: boolean + hsts: + default: true + description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. + type: boolean + hstsPreload: + default: false + description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. + type: boolean + permissionsPolicy: + default: true + description: |- + PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features + if the 'Permissions-Policy' header is not set by the upstream. + type: boolean + referrerPolicy: + default: true + description: |- + ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests + if the 'Referrer-Policy' header is not set by the upstream. + type: boolean + xContentTypeOptions: + default: true + description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. + type: boolean + xFrameOptions: + default: true + description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to add. + items: + description: HeaderRewritesHeader specifies a header with a particular value + properties: + name: + description: Name defines the name of a header. + minLength: 1 + type: string + value: + description: Value defines the value of a header. + type: string + required: + - name + - value + type: object + minItems: 1 + type: array + mode: + default: AddIfAbsent + description: Mode defines the header addition strategy. + enum: + - AddIfAbsent + - OverwriteOrAdd + type: string + name: + description: Name describing the configured operation. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + allow: + description: |- + Allow defines which response headers will be forwarded to the downstream. + This can either be allHeaders or matchingHeaders. + Default: allHeaders: {} + properties: + allHeaders: + description: AllHeaders specifies that all response headers should be forwarded. + type: object + matchingHeaders: + description: MatchingHeaders specifies which response headers should be forwarded. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response header. + properties: + standardHeaders: + default: false + description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to allow. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + remove: + description: Remove defines which response headers will be removed before forwarding to the downstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response headers. + properties: + auth: + description: Auth defines the categories of headers concerning authentication. + properties: + basic: + default: false + description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. + type: boolean + negotiate: + default: true + description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. + type: boolean + ntlm: + default: true + description: |- + NTLM removes upstream response headers that advise clients to authenticate with NTLM. + By default, these headers are removed, because NTLM pass-through is not supported. + type: boolean + type: object + informationLeakage: + description: InformationLeakage defines the categories of headers concerning information leakage. + properties: + application: + default: true + description: Application removes upstream response headers that leak information about the deployed software. + type: boolean + server: + default: true + description: Server removes upstream response headers that leak information about the server. + type: boolean + type: object + permissiveCors: + default: true + description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to remove. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured remove operation. Must be unique. + minLength: 1 + type: string + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this operation to be applied. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + settings: + description: Settings configures the HeaderRewrites filter. + properties: + operationalMode: + default: Production + description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. + enum: + - Production + - Integration + type: string + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/identitypropagations.microgateway.airlock.com.yaml similarity index 64% rename from charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/identitypropagations.microgateway.airlock.com.yaml index 8ff36ad33..a51e47545 100644 --- a/charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/identitypropagations.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: identitypropagations.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -47,6 +47,48 @@ spec: spec: description: Specification of the desired identity propagation. properties: + bearerToken: + description: BearerToken configures identity propagation via an authorization header containing a bearer token. + properties: + source: + description: Source from which to extract the token. + properties: + metadata: + description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. + properties: + key: + description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. + minLength: 1 + type: string + namespace: + description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. + minLength: 1 + type: string + required: + - key + - namespace + type: object + oidc: + description: OIDC specifies to extract a value from the result of an OpenID Connect flow. + properties: + accessToken: + description: AccessToken specifies to extract the value from the OpenID Connect Access Token. + type: object + idToken: + description: IDToken specifies to extract the value from the OpenID Connect ID Token. + properties: + claim: + description: Claim selects the JWT claim from which to extract the value. + minLength: 1 + type: string + required: + - claim + type: object + type: object + type: object + required: + - source + type: object header: description: Header configures identity propagation via a request header. properties: @@ -78,6 +120,9 @@ spec: oidc: description: OIDC specifies to extract a value from the result of an OpenID Connect flow. properties: + accessToken: + description: AccessToken specifies to extract the value from the OpenID Connect Access Token. + type: object idToken: description: IDToken specifies to extract the value from the OpenID Connect ID Token. properties: @@ -88,8 +133,6 @@ spec: required: - claim type: object - required: - - idToken type: object type: object required: diff --git a/charts/airlock/microgateway/4.4.1/crds/jwks.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/jwks.microgateway.airlock.com.yaml new file mode 100644 index 000000000..a780e6a34 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/crds/jwks.microgateway.airlock.com.yaml @@ -0,0 +1,294 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.4 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.4.1 + name: jwks.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: JWKS + listKind: JWKSList + plural: jwks + singular: jwks + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: JWKS provides a JSON Web Key Set. + 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: Specification of the JWKS. + properties: + provider: + description: Provider configures the source from which to retrieve the JWKS. + properties: + local: + description: Local specifies to retrieve the JWKS from a local secret. + properties: + secretRef: + description: SecretRef selects the secret containing the JWKS under the key 'jwks.json'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + remote: + description: Remote specifies to retrieve the JWKS from a remote endpoint. + properties: + timeouts: + description: Timeouts specifies the timeouts when interacting with the Token endpoint. + properties: + connect: + default: 5s + description: Connect specifies the timeout for establishing a connection. + type: string + maxDuration: + default: 15s + description: MaxDuration specifies the response timeout. + type: string + type: object + tls: + description: TLS defines TLS settings. + properties: + certificateVerification: + description: CertificateVerification specifies how the certificate presented by the server is verified. + properties: + custom: + description: |- + Custom explicitly specifies how the server certificate should be verified. + Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. + properties: + allowedSANs: + description: |- + AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, + that is to say, the SAN is verified if at least one matcher is matched. + AllowedSANs requires trustedCA to be set. + items: + description: |- + TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. + properties: + matcher: + description: Matcher defines the string matcher for the SAN value. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + sanType: + description: SanType defines the type of SAN matcher. + enum: + - DNS + - Email + - URI + - IPAddress + type: string + required: + - matcher + - sanType + type: object + minItems: 1 + type: array + certificatePinning: + description: |- + CertificatePinning defines constraints the presented certificate must fulfill. + If more than one constraint is configured only one must be satisfied. + At least one of allowedSPKIs and allowedHashes must be set. + properties: + allowedHashes: + description: |- + AllowedHashes is a list of hex-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + allowedSPKIs: + description: |- + AllowedSPKIs is a list of base64-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + type: object + crl: + description: CRL defines the Certificate Revocation List (CRL) settings. + properties: + lists: + description: Lists defines the list of secretRefs containing Certificate Revocation Lists. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + validationMode: + default: VerifyChain + description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. + enum: + - VerifyLeafCertOnly + - VerifyChain + type: string + type: object + trustedCA: + description: TrustedCA defines which CA certificates are trusted. + properties: + certificates: + description: Certificates defines the list of secretRefs containing trusted CA certificates. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + verificationDepth: + default: 1 + description: |- + VerificationDepth specifies the hops in the certificate chain at which validation is performed. + 1 means that either the leaf or the signing CA must be in the set of trusted certificates. + format: int32 + type: integer + required: + - certificates + type: object + type: object + disabled: + description: |- + Disabled specifies to trust any certificate without verification. + THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. + type: object + publicCAs: + description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. + type: object + type: object + ciphers: + description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. + items: + type: string + minItems: 1 + type: array + protocol: + description: Protocol defines the supported TLS protocol versions. + properties: + maximum: + description: Maximum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + minimum: + description: Minimum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + type: object + type: object + uri: + description: URI specifies the endpoint address. + format: uri + minLength: 1 + pattern: ^(http|https)://.*$ + type: string + required: + - uri + type: object + type: object + required: + - provider + type: object + required: + - spec + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/limits.microgateway.airlock.com.yaml similarity index 99% rename from charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/limits.microgateway.airlock.com.yaml index f807994db..89ba9977c 100644 --- a/charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/limits.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: limits.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/oidcproviders.microgateway.airlock.com.yaml similarity index 86% rename from charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/oidcproviders.microgateway.airlock.com.yaml index 56dad4130..9777a206b 100644 --- a/charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/oidcproviders.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: oidcproviders.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -28,19 +28,7 @@ spec: description: |- OIDCProvider specifies an OpenID Provider (OP). - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} + {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} properties: apiVersion: description: |- @@ -83,6 +71,18 @@ spec: token: description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. properties: + timeouts: + description: Timeouts specifies the timeouts when interacting with the Token endpoint. + properties: + connect: + default: 5s + description: Connect specifies the timeout for establishing a connection. + type: string + maxDuration: + default: 15s + description: MaxDuration specifies the response timeout. + type: string + type: object tls: description: TLS defines TLS settings. properties: @@ -293,8 +293,45 @@ spec: - authorization - token type: object + issuer: + description: Issuer specifies the unique identifier of the OIDC Provider, which is used e.g. for signature verification. + format: uri + minLength: 1 + pattern: ^(http|https)://.*$ + type: string + tokenValidation: + description: TokenValidation configures token validation. + properties: + idToken: + description: IDToken configures validation for the OIDC ID Token. + properties: + signatureVerification: + description: SignatureVerification specifies how to verify the ID Token signature. + properties: + disabled: + description: Disabled specifies to skip verification of the JWT signature. Not recommended for production environments. + type: object + jwksRef: + description: JwksRef specifies the JWKS to use for verifying the JWT signature (usually provided by the OpenID Provider). + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + required: + - signatureVerification + type: object + required: + - idToken + type: object required: - endpoints + - issuer + - tokenValidation type: object type: object required: diff --git a/charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml similarity index 84% rename from charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml index 578ac39df..4005300a3 100644 --- a/charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/oidcrelyingparties.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: oidcrelyingparties.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -28,19 +28,6 @@ spec: description: |- OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). - - {{% notice warning %}} The OIDC feature is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. - In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: - - The state parameter is guessable. - - Sessions are always shared across all Microgateway Engines using the same Redis instance. - I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A - may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. - - - {{% /notice %}} {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} properties: apiVersion: @@ -96,6 +83,10 @@ spec: required: - clientSecret type: object + flowTimeout: + default: 5m + description: FlowTimeout specifies the time window within which an initiated OIDC flow can be completed by the client. + type: string oidcProviderRef: description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. properties: @@ -110,7 +101,10 @@ spec: description: PathMapping configures the action matching. properties: logoutPath: - description: LogoutPath specifies which request paths should initiate a logout. + description: |- + LogoutPath specifies which request paths should initiate a logout. + + WARNING: If the AccessControl policy referencing this OIDCRelyingParty has a request condition, you must currently ensure that it also matches these logout requests. properties: matcher: description: StringMatcher defines the way to match a string. @@ -155,7 +149,10 @@ spec: - matcher type: object redirectPath: - description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. + description: |- + RedirectPath specifies which request paths should be interpreted as a callback redirect from the authorization endpoint. + + WARNING: If the AccessControl policy referencing this OIDCRelyingParty has a request condition, you must currently ensure that it also matches these callback redirect requests. properties: matcher: description: StringMatcher defines the way to match a string. @@ -206,9 +203,23 @@ spec: redirectURI: description: |- RedirectURI configures the "redirect_uri" parameter included in the authorization request. - May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. + May contain envoy command operators, e.g.: `%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback` + + WARNING: If the AccessControl policy referencing this OIDCRelyingParty has a request condition, you must currently + ensure that it also matches requests to this URI. minLength: 1 type: string + scopes: + description: |- + Scopes specifies the scopes to request during the OIDC flow. + The mandatory `openid` scope is implicitly added to the list if not already present. + Default: `['openid', 'profile']` + + Note: Different OIDCRelyingParties which use the same OIDC Provider and Client ID must request the same scopes for now. + items: + minLength: 1 + type: string + type: array required: - clientID - credentials diff --git a/charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/openapis.microgateway.airlock.com.yaml similarity index 98% rename from charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/openapis.microgateway.airlock.com.yaml index 451dc8b88..2531bc334 100644 --- a/charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/openapis.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: openapis.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.3/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/parsers.microgateway.airlock.com.yaml similarity index 99% rename from charts/airlock/microgateway/4.3.3/crds/parsers.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/parsers.microgateway.airlock.com.yaml index db60b6c84..5ed82205d 100644 --- a/charts/airlock/microgateway/4.3.3/crds/parsers.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/parsers.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.3 + app.kubernetes.io/version: 4.4.1 name: parsers.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/redisproviders.microgateway.airlock.com.yaml similarity index 67% rename from charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/redisproviders.microgateway.airlock.com.yaml index 59c94b966..65c785f0a 100644 --- a/charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/redisproviders.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: redisproviders.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -74,6 +74,63 @@ spec: mode: description: Mode configures the redis deployment mode. properties: + cluster: + description: Cluster specifies the Redis Cluster to connect to. + properties: + nodes: + description: Nodes specifies the Cluster nodes. + items: + properties: + host: + description: Host specifies the IP or hostname. + minLength: 1 + pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ + type: string + port: + default: 6379 + description: Port specifies the port. + maximum: 65535 + minimum: 1 + type: integer + required: + - host + type: object + minItems: 1 + type: array + required: + - nodes + type: object + sentinel: + description: Sentinel specifies the Redis Sentinels to connect to. + properties: + masterName: + description: MasterName specifies the master name. + minLength: 1 + type: string + nodes: + description: Nodes specifies the Sentinel nodes. + items: + properties: + host: + description: Host specifies the IP or hostname. + minLength: 1 + pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ + type: string + port: + default: 6379 + description: Port specifies the port. + maximum: 65535 + minimum: 1 + type: integer + required: + - host + type: object + minItems: 1 + type: array + required: + - masterName + - nodes + type: object standalone: description: Standalone specifies the standalone Redis instance to connect to. properties: @@ -148,6 +205,22 @@ spec: description: PublicCAs specifies to only accept certificates with a SAN matching the host and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Session Agent’s base image. type: object type: object + clientCertificate: + description: ClientCertificate configures client certificate authentication. If not specified, TLS-based client authentication is disabled. + properties: + secretRef: + description: SecretRef specifies the client certificate to use (secret of type kubernetes.io/tls). + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object type: object required: - mode diff --git a/charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/sessionhandlings.microgateway.airlock.com.yaml similarity index 57% rename from charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/sessionhandlings.microgateway.airlock.com.yaml index 5275aa7b4..81ed6ac88 100644 --- a/charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/sessionhandlings.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: sessionhandlings.microgateway.airlock.com spec: group: microgateway.airlock.com @@ -21,15 +21,7 @@ spec: - name: v1alpha1 schema: openAPIV3Schema: - description: |- - SessionHandling contains the configuration for session handling. - - - {{% notice warning %}} The Session Handling feature (required for OIDC) is currently in an experimental state. - - - We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as high-availability Redis configurations (e.g. Sentinel/Cluster) are not yet supported. - {{% /notice %}} + description: SessionHandling contains the configuration for session handling. properties: apiVersion: description: |- @@ -51,6 +43,14 @@ spec: spec: description: Specification of the desired session handling behavior. properties: + defaultTimeouts: + description: DefaultTimeouts specifies the session timeouts to apply when not provided by the authentication method. + properties: + lifetime: + default: 12h + description: Lifetime specifies the maximum duration a session can exist. + type: string + type: object persistence: description: Persistence configures where to store the session state. properties: @@ -67,6 +67,18 @@ spec: required: - redisProviderRef type: object + prefix: + description: |- + Prefix specifies the prefix under which the sessions should be stored in the persistence layer. + If not specified, an automatic prefix derived from the namespaced SessionHandling CR name is used, which ensures that sessions will always be isolated on Microgateways configured with different SessionHandling CRs, even if they share the same persistence backend. + + To allow session sharing between different Microgateway deployments, ensure that the prefix and persistence backend is the same across all corresponding SessionHandling CRs. + + Note: Session cookies are currently never shared across different fully qualified domain names (FQDNs) and authentication via different OIDC Relying Parties generates different session cookies. Clients will therefore only able to transparently reuse session cookies for connecting to different Microgateway deployments if those are a) exposed under the same FQDN and b) handle authentication via the same OIDC Relying Party. + maxLength: 64 + minLength: 1 + pattern: ^[a-zA-Z][a-zA-Z0-9_]*$ + type: string required: - persistence type: object diff --git a/charts/airlock/microgateway/4.3.1/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/sidecargateways.microgateway.airlock.com.yaml similarity index 99% rename from charts/airlock/microgateway/4.3.1/crds/sidecargateways.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/sidecargateways.microgateway.airlock.com.yaml index 9639c94fb..7229bacc0 100644 --- a/charts/airlock/microgateway/4.3.1/crds/sidecargateways.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/sidecargateways.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.1 + app.kubernetes.io/version: 4.4.1 name: sidecargateways.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.4.1/crds/telemetries.microgateway.airlock.com.yaml similarity index 98% rename from charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml rename to charts/airlock/microgateway/4.4.1/crds/telemetries.microgateway.airlock.com.yaml index 80a7cba97..96ef223f2 100644 --- a/charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml +++ b/charts/airlock/microgateway/4.4.1/crds/telemetries.microgateway.airlock.com.yaml @@ -2,10 +2,10 @@ apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.15.0 + controller-gen.kubebuilder.io/version: v0.16.4 labels: app.kubernetes.io/name: airlock-microgateway-operator - app.kubernetes.io/version: 4.3.0 + app.kubernetes.io/version: 4.4.1 name: telemetries.microgateway.airlock.com spec: group: microgateway.airlock.com diff --git a/charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json b/charts/airlock/microgateway/4.4.1/dashboards/blockLogs.json similarity index 68% rename from charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json rename to charts/airlock/microgateway/4.4.1/dashboards/blockLogs.json index ef0ce6d62..8c96b4f64 100644 --- a/charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/blockLogs.json @@ -60,7 +60,7 @@ } ] }, - "description": "Blocked requests by Airlock Microgateway retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for even a more granular filtering of the logs.", + "description": "Log entries of threats blocked by Airlock Microgateway.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for an even more granular filtering of the logs.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -140,7 +140,7 @@ }, { "id": "unit", - "value": "dateTimeAsIso" + "value": "time: YYYY-MM-DD HH:mm:ss.SSS" }, { "id": "custom.filterable" @@ -150,12 +150,12 @@ { "matcher": { "id": "byName", - "options": "Method" + "options": "HTTP Method" }, "properties": [ { "id": "custom.width", - "value": 89 + "value": 140 } ] }, @@ -222,7 +222,7 @@ { "matcher": { "id": "byName", - "options": "Attack Type" + "options": "Block Subtype" }, "properties": [ { @@ -230,18 +230,6 @@ "value": 217 } ] - }, - { - "matcher": { - "id": "byName", - "options": "Application" - }, - "properties": [ - { - "id": "custom.width", - "value": 207 - } - ] } ] }, @@ -266,7 +254,7 @@ "showHeader": true, "sortBy": [] }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -274,62 +262,14 @@ "uid": "${DS_LOKI}" }, "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_deny_rule\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.deny_rules.matches\"\n| label_format block_type=\"deny_rules\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule_key }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", domain=\"url.domain\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.actions.block.details\", block_type=\"airlock.actions.block.block_type\", block_subtype=\"airlock.actions.block.block_subtype\"\n| block_type=~\"${blockType:regex}\"", "hide": false, "queryType": "range", - "refId": "Deny Rule Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_limit\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.limits.matches\"\n| label_format block_type=\"limits\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Limit Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_openapi\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.openapi.reference\", constraint=\"airlock.openapi.request.failed_validation.constraint\", position=\"airlock.openapi.request.failed_validation.position\", message=\"airlock.openapi.request.failed_validation.message\"\n| label_format block_type=\"openapi\", attack_type=\"openapi\", details=`{{.reference }}: {{.constraint }} at {{ .position }} ({{ .message }})` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "OpenAPI Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_parser\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", attack_type=\"airlock.parser\", failed_check=\"airlock.parser.matches[0].failed_check\", message=\"airlock.parser.matches[0].message\"\n| label_format block_type=\"parsing\", attack_type=\"parsing\", details=`{{.failed_check}}: {{.message}}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Parser Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_graphql\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.graphql.reference\", message=\"airlock.graphql.request.failed_validation.message\"\n| label_format block_type=\"graphql\", attack_type=\"graphql\", details=`{{ .reference }}: {{ .message }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "GraphQL Blocks" + "refId": "Blocks" } ], "title": "Blocked Request logs", "transformations": [ - { - "id": "merge", - "options": {} - }, { "id": "extractFields", "options": { @@ -344,16 +284,16 @@ "include": { "names": [ "Time", - "attack_type", + "block_subtype", "block_type", "client_ip", "details", + "domain", "http_method", "namespace", "request_id", "request_size", - "url", - "pod" + "url" ] } } @@ -371,30 +311,29 @@ "includeByName": {}, "indexByName": { "Time": 0, - "attack_type": 7, + "block_subtype": 7, "block_type": 6, "client_ip": 9, "details": 8, + "domain": 2, "http_method": 3, "namespace": 1, - "pod": 2, "request_id": 10, "request_size": 5, "url": 4 }, "renameByName": { "Time": "Timestamp", - "attack_type": "Attack Type", + "block_subtype": "Block Subtype", "block_type": "Block Type", "client_ip": "Client IP", "details": "Details", - "http_method": "Method", + "domain": "URL Domain", + "http_method": "HTTP Method", "namespace": "Namespace", - "pod": "Pod", "request_id": "Request ID", "request_size": "Request Size", - "tsNs": "", - "url": "Path" + "url": "URL Path" } } } @@ -409,11 +348,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_LOKI", @@ -477,11 +412,7 @@ "type": "query" }, { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_PROMETHEUS", @@ -503,7 +434,7 @@ "timeRangeUpdatedDuringEditOrView": false, "timepicker": {}, "timezone": "browser", - "title": "Airlock Microgateway Blocked Request Logs", + "title": "Airlock Microgateway Threats Block - Logs", "uid": "adnyzcvwnyadcc", "version": 3, "weekStart": "" diff --git a/charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json b/charts/airlock/microgateway/4.4.1/dashboards/blockMetrics.json similarity index 95% rename from charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json rename to charts/airlock/microgateway/4.4.1/dashboards/blockMetrics.json index ba383d22e..0b98122ef 100644 --- a/charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/blockMetrics.json @@ -58,7 +58,7 @@ } ] }, - "description": "Metrics on requests blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", + "description": "Metrics on threats blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -88,7 +88,7 @@ "y": 0 }, "id": 6, - "title": "Airlock Microgateway Block Metrics", + "title": "Airlock Microgateway Threats Block - Metrics", "type": "row" }, { @@ -140,7 +140,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -225,7 +225,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -408,7 +408,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Blocked requests by block type.", + "description": "Blocked threats by block type.", "fieldConfig": { "defaults": { "color": { @@ -448,7 +448,7 @@ } ] }, - "unit": "none" + "unit": "short" }, "overrides": [] }, @@ -482,7 +482,7 @@ "xTickLabelRotation": 0, "xTickLabelSpacing": 0 }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -520,7 +520,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Blocked requests by attack type, which are subsets of the various block types.", + "description": "Blocked threats by block subtype, which are subsets of the various block types.", "fieldConfig": { "defaults": { "color": { @@ -557,7 +557,8 @@ "value": null } ] - } + }, + "unit": "short" }, "overrides": [] }, @@ -587,11 +588,11 @@ "mode": "single", "sort": "none" }, - "xField": "attack_type", + "xField": "block_subtype", "xTickLabelRotation": 0, "xTickLabelSpacing": 0 }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -600,14 +601,14 @@ }, "editorMode": "code", "exemplar": false, - "expr": "round(sum by (attack_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "expr": "round(sum by (block_subtype) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", "instant": true, "legendFormat": "__auto", "range": false, "refId": "A" } ], - "title": "Attack Type", + "title": "Block Subtype", "transformations": [ { "id": "reduce", @@ -630,11 +631,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "Datasource Prometheus", @@ -648,11 +645,7 @@ "type": "datasource" }, { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_LOKI", @@ -751,7 +744,7 @@ "hidden": false }, "timezone": "browser", - "title": "Airlock Microgateway Block Metrics", + "title": "Airlock Microgateway Threats Block - Metrics", "uid": "ddnqoczu7qvb4cdd3dd", "version": 3, "weekStart": "" diff --git a/charts/airlock/microgateway/4.3.3/dashboards/blockLogs.json b/charts/airlock/microgateway/4.4.1/dashboards/headerLogs.json similarity index 55% rename from charts/airlock/microgateway/4.3.3/dashboards/blockLogs.json rename to charts/airlock/microgateway/4.4.1/dashboards/headerLogs.json index ef0ce6d62..a6c45008f 100644 --- a/charts/airlock/microgateway/4.3.3/dashboards/blockLogs.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/headerLogs.json @@ -7,14 +7,6 @@ "type": "datasource", "pluginId": "loki", "pluginName": "Loki" - }, - { - "name": "DS_PROMETHEUS", - "label": "Prometheus", - "description": "", - "type": "datasource", - "pluginId": "prometheus", - "pluginName": "Prometheus" } ], "__elements": {}, @@ -60,7 +52,7 @@ } ] }, - "description": "Blocked requests by Airlock Microgateway retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for even a more granular filtering of the logs.", + "description": "Logs for header rewrites by Airlock Microgateway, retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace. Column filters on the table allow for an even more granular filtering of the logs.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -84,6 +76,7 @@ "panels": [ { "datasource": { + "default": false, "type": "loki", "uid": "${DS_LOKI}" }, @@ -140,7 +133,7 @@ }, { "id": "unit", - "value": "dateTimeAsIso" + "value": "time: YYYY-MM-DD HH:mm:ss.SSS" }, { "id": "custom.filterable" @@ -150,12 +143,12 @@ { "matcher": { "id": "byName", - "options": "Method" + "options": "HTTP Method" }, "properties": [ { "id": "custom.width", - "value": 89 + "value": 140 } ] }, @@ -183,22 +176,6 @@ } ] }, - { - "matcher": { - "id": "byName", - "options": "Block Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 116 - }, - { - "id": "custom.filterable", - "value": false - } - ] - }, { "matcher": { "id": "byName", @@ -218,30 +195,6 @@ "value": "right" } ] - }, - { - "matcher": { - "id": "byName", - "options": "Attack Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 217 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Application" - }, - "properties": [ - { - "id": "custom.width", - "value": 207 - } - ] } ] }, @@ -266,7 +219,7 @@ "showHeader": true, "sortBy": [] }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -274,62 +227,14 @@ "uid": "${DS_LOKI}" }, "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_deny_rule\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.deny_rules.matches\"\n| label_format block_type=\"deny_rules\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule_key }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"header_rewrites\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", domain=\"url.domain\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", header_request_details=\"airlock.actions.header_rewrites.request\", header_response_details=\"airlock.actions.header_rewrites.response\", log_type=\"event.dataset\" | log_type = `envoy.access`", "hide": false, "queryType": "range", - "refId": "Deny Rule Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_limit\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.limits.matches\"\n| label_format block_type=\"limits\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Limit Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_openapi\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.openapi.reference\", constraint=\"airlock.openapi.request.failed_validation.constraint\", position=\"airlock.openapi.request.failed_validation.position\", message=\"airlock.openapi.request.failed_validation.message\"\n| label_format block_type=\"openapi\", attack_type=\"openapi\", details=`{{.reference }}: {{.constraint }} at {{ .position }} ({{ .message }})` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "OpenAPI Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_parser\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", attack_type=\"airlock.parser\", failed_check=\"airlock.parser.matches[0].failed_check\", message=\"airlock.parser.matches[0].message\"\n| label_format block_type=\"parsing\", attack_type=\"parsing\", details=`{{.failed_check}}: {{.message}}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Parser Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_graphql\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.graphql.reference\", message=\"airlock.graphql.request.failed_validation.message\"\n| label_format block_type=\"graphql\", attack_type=\"graphql\", details=`{{ .reference }}: {{ .message }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "GraphQL Blocks" + "refId": "Header Rewrites" } ], - "title": "Blocked Request logs", + "title": "Header Rewrite Logs", "transformations": [ - { - "id": "merge", - "options": {} - }, { "id": "extractFields", "options": { @@ -344,16 +249,15 @@ "include": { "names": [ "Time", - "attack_type", - "block_type", "client_ip", - "details", + "domain", + "header_request_details", + "header_response_details", "http_method", "namespace", "request_id", "request_size", - "url", - "pod" + "url" ] } } @@ -371,30 +275,28 @@ "includeByName": {}, "indexByName": { "Time": 0, - "attack_type": 7, - "block_type": 6, - "client_ip": 9, - "details": 8, + "client_ip": 8, + "domain": 2, + "header_request_details": 6, + "header_response_details": 7, "http_method": 3, "namespace": 1, - "pod": 2, - "request_id": 10, + "request_id": 9, "request_size": 5, "url": 4 }, "renameByName": { "Time": "Timestamp", - "attack_type": "Attack Type", - "block_type": "Block Type", "client_ip": "Client IP", "details": "Details", - "http_method": "Method", + "domain": "URL Domain", + "header_request_details": "Request Header Actions", + "header_response_details": "Response Header Actions", + "http_method": "HTTP Method", "namespace": "Namespace", - "pod": "Pod", "request_id": "Request ID", "request_size": "Request Size", - "tsNs": "", - "url": "Path" + "url": "URL Path" } } } @@ -409,11 +311,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_LOKI", @@ -452,36 +350,7 @@ "type": "query" }, { - "allValue": ".*", "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "hide": 0, - "includeAll": true, - "label": "Block Type", - "multi": true, - "name": "blockType", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, "hide": 2, "includeAll": false, "label": "DS_PROMETHEUS", @@ -500,11 +369,10 @@ "from": "now-15m", "to": "now" }, - "timeRangeUpdatedDuringEditOrView": false, "timepicker": {}, "timezone": "browser", - "title": "Airlock Microgateway Blocked Request Logs", - "uid": "adnyzcvwnyadcc", - "version": 3, + "title": "Airlock Microgateway Header Rewrites - Logs", + "uid": "adnydadenyadcc", + "version": 1, "weekStart": "" } \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.4/dashboards/overview.json b/charts/airlock/microgateway/4.4.1/dashboards/license.json similarity index 54% rename from charts/airlock/microgateway/4.3.4/dashboards/overview.json rename to charts/airlock/microgateway/4.4.1/dashboards/license.json index 094276621..14886328a 100644 --- a/charts/airlock/microgateway/4.3.4/dashboards/overview.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/license.json @@ -58,6 +58,7 @@ } ] }, + "description": "Overview on Airlock Microgateway License attributes and usage.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -79,246 +80,12 @@ } ], "panels": [ - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 0 - }, - "id": 3, - "title": "Overview", - "type": "row" - }, { "datasource": { "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Number of pods that are protected by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "thresholds" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "text", - "value": null - } - ] - } - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 0, - "y": 1 - }, - "id": 11, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "exemplar": false, - "expr": "sum(microgateway_sidecars{namespace=~\"${operator_namespace.regex}\"})", - "instant": true, - "legendFormat": "Protected Pods", - "range": false, - "refId": "A" - } - ], - "title": "Protected Pods", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 3, - "y": 1 - }, - "id": 4, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [ - { - "options": { - "match": "nan", - "result": { - "index": 0, - "text": "n/a" - } - }, - "type": "special" - } - ], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "percentunit" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 3, - "x": 6, - "y": 1 - }, - "id": 5, - "options": { - "colorMode": "value", - "graphMode": "area", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "last" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", - "fullMetaSearch": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Blocked Requests (%)", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "% Blocked Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "License status of Airlock Microgateway.", + "description": "Aggregated status of the Airlock Microgateway licenses selected in the dashboard filter.", "fieldConfig": { "defaults": { "color": { @@ -356,15 +123,16 @@ "gridPos": { "h": 4, "w": 3, - "x": 9, - "y": 1 + "x": 0, + "y": 0 }, - "id": 10, + "id": 1, "options": { "colorMode": "value", "graphMode": "area", "justifyMode": "auto", "orientation": "auto", + "percentChangeColorMode": "standard", "reduceOptions": { "calcs": [ "lastNotNull" @@ -376,7 +144,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -385,33 +153,229 @@ }, "editorMode": "code", "exemplar": false, - "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", + "expr": "min(microgateway_license_valid * on (service,instance) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})", "instant": true, "legendFormat": "License Status", "range": false, "refId": "Licenses" } ], - "title": "License", + "title": "License Status", "type": "stat" }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 5 - }, - "id": 2, - "title": "Blocks", - "type": "row" - }, { "datasource": { "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", + "description": "Next upcoming expiry date over all Airlock Microgateway licenses selected in the dashboard filter.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "time: L" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 3, + "y": 0 + }, + "id": 4, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "10.2.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "min(microgateway_license_expiry_timestamp_seconds * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})*1000", + "instant": true, + "legendFormat": "Expiry Date (MM/DD/YYYY)", + "range": false, + "refId": "A" + } + ], + "title": "License Expiry Date", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Sum of the number licensed requests over all Airlock Microgateway license selected in the dashboard filter.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 7, + "y": 0 + }, + "id": 6, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "10.2.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(topk(1, (microgateway_license_max_rq_count_per_month > 0) * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"}) by (id))", + "instant": true, + "legendFormat": "Licensed Requests", + "range": false, + "refId": "A" + } + ], + "title": "Licensed Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Sum of the estimated number of requests over 30 days based on the last 7 days over all Airlock Microgateway licenses selected in the dashboard filter.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 5, + "x": 11, + "y": 0 + }, + "id": 2, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "10.2.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "(sum((label_replace(increase(microgateway_license_http_rq_total[7d]), \"namespace\", \"$1\", \"job\", \"(.+)/.*\")) * on(namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"}))/7*30", + "instant": true, + "legendFormat": "Estimated Requests", + "range": false, + "refId": "A" + } + ], + "title": "Requests over 30 days (estimated)", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Number of requests per week processed by Airlock Microgateway.", "fieldConfig": { "defaults": { "color": { @@ -423,8 +387,9 @@ "axisCenteredZero": false, "axisColorMode": "text", "axisLabel": "", - "axisPlacement": "left", + "axisPlacement": "auto", "barAlignment": 0, + "barWidthFactor": 0.6, "drawStyle": "line", "fillOpacity": 0, "gradientMode": "none", @@ -455,65 +420,22 @@ "mode": "absolute", "steps": [ { - "color": "blue", + "color": "green", "value": null } ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] + "unit": "short" + }, + "overrides": [] }, "gridPos": { - "h": 10, - "w": 12, + "h": 13, + "w": 24, "x": 0, - "y": 6 + "y": 4 }, - "id": 6, + "id": 5, "options": { "legend": { "calcs": [], @@ -521,9 +443,6 @@ "placement": "bottom", "showLegend": true }, - "timezone": [ - "" - ], "tooltip": { "maxHeight": 600, "mode": "single", @@ -537,36 +456,24 @@ "uid": "${DS_PROMETHEUS}" }, "editorMode": "code", - "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "instant": false, - "legendFormat": "Requests per second", - "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "expr": " sum((label_replace(avg_over_time(increase(microgateway_license_http_rq_total[7d])[2m:30s]), \"namespace\", \"$1\", \"job\", \"(.+)/.*\")) * on(namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})", "hide": false, "instant": false, - "legendFormat": "% Blocks", + "legendFormat": "# Requests per week", "range": true, - "refId": "Blocks" + "refId": "C" } ], - "title": "Requests vs. % Blocks", + "title": "Processed Requests per week", "type": "timeseries" }, { "datasource": { + "default": false, "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Requests blocked by Airlock Microgateway categorized by their corresponding type.", + "description": "Estimated number of requests over 30 days based on the last 7 days per operator namespace for the Airlock Microgateway licenses selected in the dashboard filter.", "fieldConfig": { "defaults": { "color": { @@ -575,26 +482,10 @@ "custom": { "align": "auto", "cellOptions": { - "barAlignment": 0, - "drawStyle": "line", - "gradientMode": "none", - "hideValue": false, - "lineInterpolation": "linear", - "lineStyle": { - "dash": [ - 10, - 10 - ], - "fill": "solid" - }, - "showPoints": "never", - "spanNulls": false, - "type": "sparkline" + "type": "auto" }, "inspect": false }, - "displayName": "Block Type", - "fieldMinMax": false, "mappings": [], "thresholds": { "mode": "absolute", @@ -602,6 +493,10 @@ { "color": "green", "value": null + }, + { + "color": "red", + "value": 80 } ] } @@ -610,67 +505,78 @@ { "matcher": { "id": "byName", - "options": "block_type" + "options": "License ID" }, "properties": [ { "id": "custom.width", - "value": 153 - }, - { - "id": "custom.cellOptions", - "value": { - "type": "auto" - } + "value": 330 } ] }, { "matcher": { "id": "byName", - "options": "Trend #Block Types" + "options": "Requests over 30 days (estimated)" }, "properties": [ { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } + "id": "unit", + "value": "short" + }, + { + "id": "mappings", + "value": [ + { + "options": { + "match": "null+nan", + "result": { + "index": 0, + "text": "0" + } + }, + "type": "special" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Operator Namespace" + }, + "properties": [ + { + "id": "custom.width", + "value": 307 } ] } ] }, "gridPos": { - "h": 10, - "w": 12, - "x": 12, - "y": 6 + "h": 8, + "w": 11, + "x": 0, + "y": 17 }, "id": 7, "options": { - "cellHeight": "lg", + "cellHeight": "sm", "footer": { "countRows": false, - "enablePagination": false, - "fields": [ - "Value" - ], + "fields": "", "reducer": [ "sum" ], "show": false }, - "showHeader": false, - "sortBy": [ - { - "desc": true, - "displayName": "block_type" - } - ] + "frameIndex": 1, + "showHeader": true, + "sortBy": [] }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -678,84 +584,91 @@ "uid": "${DS_PROMETHEUS}" }, "editorMode": "code", - "expr": "sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m] offset -1m))/(60000/$__interval_ms)", - "format": "time_series", - "instant": false, + "exemplar": false, + "expr": "(sum by (namespace, id) ((label_replace(increase(microgateway_license_http_rq_total[7d]), \"namespace\", \"$1\", \"job\", \"(.+)/.*\")) * on(namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"}))/7*30", + "format": "table", + "hide": false, + "instant": true, "legendFormat": "__auto", - "range": true, - "refId": "Block Types" + "range": false, + "refId": "Est. Usage over 30 days" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "(min by(namespace) (microgateway_build_info{container=\"manager\"})) * on (namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"}", + "format": "table", + "hide": false, + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "Engine License" } ], - "title": "Blocked Requests by Type", + "title": "Usage by Operator Namespace", "transformations": [ { - "id": "timeSeriesTable", + "id": "merge", + "options": {} + }, + { + "id": "organize", "options": { - "A": { - "timeField": "Time" + "excludeByName": { + "Time": true, + "Value #Engine License": true, + "Value #Licensed Req": false, + "container": true, + "endpoint": true, + "instance": true, + "job": true, + "namespace": false, + "pod": true, + "service": true, + "version": true }, - "Block Types": { - "stat": "sum", - "timeField": "Time" + "includeByName": {}, + "indexByName": { + "Time": 0, + "Value": 3, + "id": 2, + "namespace": 1 + }, + "renameByName": { + "Value #Est. Usage over 30 days": "Requests over 30 days (estimated)", + "Value #License Expiry Date": "Expiry Date", + "Value #License Type": "License Type", + "Value #Licensed Req": "Licensed Requests", + "Value #Validity": "Valid", + "id": "License ID", + "namespace": "Operator Namespace" } } } ], "type": "table" }, - { - "gridPos": { - "h": 1, - "w": 24, - "x": 0, - "y": 16 - }, - "id": 1, - "title": "Latency", - "type": "row" - }, { "datasource": { "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Percentiles of the application downstream latency over one minute.", + "description": "Metadata for the Airlock Microgateway licenses selected in the dashboard filter.", "fieldConfig": { "defaults": { "color": { - "mode": "palette-classic" + "mode": "thresholds" }, "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false + "align": "auto", + "cellOptions": { + "type": "auto" }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } + "inspect": false }, "mappings": [], "thresholds": { @@ -764,245 +677,198 @@ { "color": "green", "value": null + }, + { + "color": "red", + "value": 80 } ] - }, - "unit": "ms" + } }, "overrides": [ { "matcher": { "id": "byName", - "options": "25th Percentile" + "options": "License ID" }, "properties": [ { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } + "id": "custom.width", + "value": 321 } ] }, { "matcher": { "id": "byName", - "options": "50th Percentile" + "options": "Valid" }, "properties": [ { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } + "id": "mappings", + "value": [ + { + "options": { + "0": { + "color": "red", + "index": 1, + "text": "Invalid" + }, + "1": { + "color": "green", + "index": 0, + "text": "Valid" + } + }, + "type": "value" + }, + { + "options": { + "match": "null+nan", + "result": { + "color": "red", + "index": 2, + "text": "Invalid" + } + }, + "type": "special" + } + ] + }, + { + "id": "custom.width", + "value": 65 } ] }, { "matcher": { "id": "byName", - "options": "95th Percentile" + "options": "License Type" }, "properties": [ { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } + "id": "mappings", + "value": [ + { + "options": { + "0": { + "index": 1, + "text": "Community" + }, + "1": { + "index": 0, + "text": "Premium" + } + }, + "type": "value" + }, + { + "options": { + "match": "null+nan", + "result": { + "index": 2, + "text": "n/a" + } + }, + "type": "special" + } + ] + }, + { + "id": "custom.width", + "value": 109 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Expiry Date" + }, + "properties": [ + { + "id": "unit", + "value": "time:L" + }, + { + "id": "custom.width", + "value": 130 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Requests over 30 days (estimated)" + }, + "properties": [ + { + "id": "unit", + "value": "short" + }, + { + "id": "mappings", + "value": [ + { + "options": { + "match": "null+nan", + "result": { + "index": 0, + "text": "0" + } + }, + "type": "special" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Licensed Requests" + }, + "properties": [ + { + "id": "unit", + "value": "short" + }, + { + "id": "custom.width", + "value": 160 } ] } ] }, "gridPos": { - "h": 9, - "w": 12, - "x": 0, + "h": 8, + "w": 13, + "x": 11, "y": 17 }, "id": 8, "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true + "cellHeight": "sm", + "footer": { + "countRows": false, + "fields": "", + "reducer": [ + "sum" + ], + "show": false }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "25th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "50th Percentile" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "95th Percentile" - } - ], - "title": "Application Downstream Latency", - "type": "timeseries" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Percentiles of the Airlock Microgateway processing time over one minute.", - "fieldConfig": { - "defaults": { - "color": { - "mode": "palette-classic" - }, - "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 0, - "gradientMode": "none", - "hideFrom": { - "legend": false, - "tooltip": false, - "viz": false - }, - "insertNulls": false, - "lineInterpolation": "linear", - "lineWidth": 1, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" - } - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "ms" - }, - "overrides": [ + "frameIndex": 1, + "showHeader": true, + "sortBy": [ { - "matcher": { - "id": "byName", - "options": "25th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "50th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "purple", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "95th Percentile" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "dark-purple", - "mode": "fixed" - } - } - ] + "desc": false, + "displayName": "Expiry Date" } ] }, - "gridPos": { - "h": 9, - "w": 12, - "x": 12, - "y": 17 - }, - "id": 9, - "options": { - "legend": { - "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true - }, - "tooltip": { - "maxHeight": 600, - "mode": "single", - "sort": "none" - } - }, + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -1010,11 +876,13 @@ "uid": "${DS_PROMETHEUS}" }, "editorMode": "code", - "expr": "histogram_quantile(0.25, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", - "instant": false, - "legendFormat": "25th Percentile", - "range": true, - "refId": "0.25 Percentile" + "exemplar": false, + "expr": "min by (id) (microgateway_license_valid * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})", + "format": "table", + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "Validity" }, { "datasource": { @@ -1022,12 +890,14 @@ "uid": "${DS_PROMETHEUS}" }, "editorMode": "code", - "expr": "histogram_quantile(0.5, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "exemplar": false, + "expr": "topk(1,microgateway_license_max_rq_count_per_month * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})by (id)", + "format": "table", "hide": false, - "instant": false, - "legendFormat": "50th Percentile", - "range": true, - "refId": "0.5 Percentile" + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "Licensed Req" }, { "datasource": { @@ -1035,16 +905,99 @@ "uid": "${DS_PROMETHEUS}" }, "editorMode": "code", - "expr": "histogram_quantile(0.95, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "exemplar": false, + "expr": "min by (id) (microgateway_license_is_premium * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})", + "format": "table", "hide": false, - "instant": false, - "legendFormat": "95th Percentile", - "range": true, - "refId": "0.95 Percentile" + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "License Type" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "min by (id) (microgateway_license_expiry_timestamp_seconds * on (service, namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"})*1000", + "format": "table", + "hide": false, + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "License Expiry Date" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "(sum by (id) ((label_replace(increase(microgateway_license_http_rq_total[7d]), \"namespace\", \"$1\", \"job\", \"(.+)/.*\")) * on(namespace) group_left(id) microgateway_license_info{id=~\"${license_id.regex}\"}))/7*30", + "format": "table", + "hide": false, + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "Est. Usage over 30 days" } ], - "title": "Airlock Microgateway Processing Time", - "type": "timeseries" + "title": "License Overview", + "transformations": [ + { + "id": "merge", + "options": {} + }, + { + "id": "organize", + "options": { + "excludeByName": { + "Time": true, + "Value #Licensed Req": false, + "container": true, + "endpoint": true, + "instance": true, + "job": true, + "namespace": true, + "pod": true, + "service": true + }, + "includeByName": {}, + "indexByName": {}, + "renameByName": { + "Value #Est. Usage over 30 days": "Requests over 30 days (estimated)", + "Value #License Expiry Date": "Expiry Date", + "Value #License Type": "License Type", + "Value #Licensed Req": "Licensed Requests", + "Value #Validity": "Valid", + "id": "License ID", + "namespace": "Operator Namespace" + } + } + }, + { + "id": "filterByValue", + "options": { + "filters": [ + { + "config": { + "id": "equal", + "options": { + "value": "" + } + }, + "fieldName": "License ID" + } + ], + "match": "any", + "type": "exclude" + } + } + ], + "type": "table" } ], "refresh": "", @@ -1055,11 +1008,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_PROMETHEUS", @@ -1079,41 +1028,17 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "definition": "label_values(microgateway_license_valid,namespace)", + "definition": "label_values(microgateway_license_info,id)", + "description": "", "hide": 0, "includeAll": true, - "label": "Operator Namespace", + "label": "License ID", "multi": true, - "name": "operator_namespace", + "name": "license_id", "options": [], "query": { "qryType": 1, - "query": "label_values(microgateway_license_valid,namespace)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": ".*", - "skipUrlSync": false, - "sort": 0, - "type": "query" - }, - { - "allValue": ".*", - "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_license_http_rq_total,namespace)", - "hide": 0, - "includeAll": true, - "label": "Application Namespace", - "multi": true, - "name": "namespace", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_license_http_rq_total,namespace)", + "query": "label_values(microgateway_license_info,id)", "refId": "PrometheusVariableQueryEditor-VariableQuery" }, "refresh": 2, @@ -1125,14 +1050,14 @@ ] }, "time": { - "from": "now-24h", + "from": "now-7d", "to": "now" }, "timeRangeUpdatedDuringEditOrView": false, "timepicker": {}, "timezone": "browser", - "title": "Airlock Microgateway Overview", - "uid": "fdp5jb8fnrmyoa", - "version": 1, + "title": "Airlock Microgateway License", + "uid": "cdpq79bzrr01se", + "version": 2, "weekStart": "" } \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/dashboards/blockLogs.json b/charts/airlock/microgateway/4.4.1/dashboards/logOnlyLogs.json similarity index 55% rename from charts/airlock/microgateway/4.3.1/dashboards/blockLogs.json rename to charts/airlock/microgateway/4.4.1/dashboards/logOnlyLogs.json index ef0ce6d62..6d7ae7f22 100644 --- a/charts/airlock/microgateway/4.3.1/dashboards/blockLogs.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/logOnlyLogs.json @@ -60,7 +60,7 @@ } ] }, - "description": "Blocked requests by Airlock Microgateway retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for even a more granular filtering of the logs.", + "description": "Log entries of threats logged in log-only mode by Airlock Microgateway.\n\nThe dashboard can be filtered by namespace. Column filters on the table allow for an even more granular filtering of the logs.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -84,6 +84,7 @@ "panels": [ { "datasource": { + "default": false, "type": "loki", "uid": "${DS_LOKI}" }, @@ -121,7 +122,7 @@ "properties": [ { "id": "custom.width", - "value": 221 + "value": 328 }, { "id": "custom.filterable" @@ -136,11 +137,11 @@ "properties": [ { "id": "custom.width", - "value": 214 + "value": 176 }, { "id": "unit", - "value": "dateTimeAsIso" + "value": "time: YYYY-MM-DD HH:mm:ss.SSS" }, { "id": "custom.filterable" @@ -150,12 +151,12 @@ { "matcher": { "id": "byName", - "options": "Method" + "options": "HTTP Method" }, "properties": [ { "id": "custom.width", - "value": 89 + "value": 132 } ] }, @@ -167,7 +168,7 @@ "properties": [ { "id": "custom.width", - "value": 138 + "value": 137 } ] }, @@ -183,22 +184,6 @@ } ] }, - { - "matcher": { - "id": "byName", - "options": "Block Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 116 - }, - { - "id": "custom.filterable", - "value": false - } - ] - }, { "matcher": { "id": "byName", @@ -218,30 +203,6 @@ "value": "right" } ] - }, - { - "matcher": { - "id": "byName", - "options": "Attack Type" - }, - "properties": [ - { - "id": "custom.width", - "value": 217 - } - ] - }, - { - "matcher": { - "id": "byName", - "options": "Application" - }, - "properties": [ - { - "id": "custom.width", - "value": 207 - } - ] } ] }, @@ -266,7 +227,7 @@ "showHeader": true, "sortBy": [] }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -274,62 +235,14 @@ "uid": "${DS_LOKI}" }, "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_deny_rule\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.deny_rules.matches\"\n| label_format block_type=\"deny_rules\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule_key }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= `log_only` |= `envoy.access` | json http_method=\"http.request.method\", url=\"url.path\", domain=\"url.domain\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.actions.log_only\", log_type=\"event.dataset\" | label_format log_count=`{{ len (fromJson .details) }}` | log_type = `envoy.access` | log_count > 0", "hide": false, "queryType": "range", - "refId": "Deny Rule Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_limit\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.limits.matches\"\n| label_format block_type=\"limits\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Limit Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_openapi\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.openapi.reference\", constraint=\"airlock.openapi.request.failed_validation.constraint\", position=\"airlock.openapi.request.failed_validation.position\", message=\"airlock.openapi.request.failed_validation.message\"\n| label_format block_type=\"openapi\", attack_type=\"openapi\", details=`{{.reference }}: {{.constraint }} at {{ .position }} ({{ .message }})` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "OpenAPI Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_parser\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", attack_type=\"airlock.parser\", failed_check=\"airlock.parser.matches[0].failed_check\", message=\"airlock.parser.matches[0].message\"\n| label_format block_type=\"parsing\", attack_type=\"parsing\", details=`{{.failed_check}}: {{.message}}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "Parser Blocks" - }, - { - "datasource": { - "type": "loki", - "uid": "${DS_LOKI}" - }, - "editorMode": "code", - "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_graphql\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.graphql.reference\", message=\"airlock.graphql.request.failed_validation.message\"\n| label_format block_type=\"graphql\", attack_type=\"graphql\", details=`{{ .reference }}: {{ .message }}` | block_type=~\"${blockType:regex}\"", - "hide": false, - "queryType": "range", - "refId": "GraphQL Blocks" + "refId": "Log Only Logs" } ], - "title": "Blocked Request logs", + "title": "Threats Logs Log-Only", "transformations": [ - { - "id": "merge", - "options": {} - }, { "id": "extractFields", "options": { @@ -344,16 +257,14 @@ "include": { "names": [ "Time", - "attack_type", - "block_type", "client_ip", "details", + "domain", "http_method", "namespace", "request_id", "request_size", - "url", - "pod" + "url" ] } } @@ -371,30 +282,25 @@ "includeByName": {}, "indexByName": { "Time": 0, - "attack_type": 7, - "block_type": 6, - "client_ip": 9, - "details": 8, - "http_method": 3, + "client_ip": 8, + "details": 7, + "domain": 2, + "http_method": 4, "namespace": 1, - "pod": 2, - "request_id": 10, - "request_size": 5, - "url": 4 + "request_id": 9, + "request_size": 6, + "url": 5 }, "renameByName": { "Time": "Timestamp", - "attack_type": "Attack Type", - "block_type": "Block Type", "client_ip": "Client IP", "details": "Details", - "http_method": "Method", + "domain": "URL Domain", + "http_method": "HTTP Method", "namespace": "Namespace", - "pod": "Pod", "request_id": "Request ID", "request_size": "Request Size", - "tsNs": "", - "url": "Path" + "url": "URL Path" } } } @@ -409,11 +315,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_LOKI", @@ -452,36 +354,7 @@ "type": "query" }, { - "allValue": ".*", "current": {}, - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "hide": 0, - "includeAll": true, - "label": "Block Type", - "multi": true, - "name": "blockType", - "options": [], - "query": { - "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", - "refId": "PrometheusVariableQueryEditor-VariableQuery" - }, - "refresh": 2, - "regex": "", - "skipUrlSync": false, - "sort": 5, - "type": "query" - }, - { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, "hide": 2, "includeAll": false, "label": "DS_PROMETHEUS", @@ -500,11 +373,10 @@ "from": "now-15m", "to": "now" }, - "timeRangeUpdatedDuringEditOrView": false, "timepicker": {}, "timezone": "browser", - "title": "Airlock Microgateway Blocked Request Logs", - "uid": "adnyzcvwnyadcc", - "version": 3, + "title": "Airlock Microgateway Threats LogOnly - Logs", + "uid": "adnasdfdwnyadcc", + "version": 7, "weekStart": "" } \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.1/dashboards/blockMetrics.json b/charts/airlock/microgateway/4.4.1/dashboards/logOnlyMetrics.json similarity index 70% rename from charts/airlock/microgateway/4.3.1/dashboards/blockMetrics.json rename to charts/airlock/microgateway/4.4.1/dashboards/logOnlyMetrics.json index ba383d22e..137e28ee0 100644 --- a/charts/airlock/microgateway/4.3.1/dashboards/blockMetrics.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/logOnlyMetrics.json @@ -58,7 +58,7 @@ } ] }, - "description": "Metrics on requests blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", + "description": "Metrics on threats logged by Airlock Microgateway in threat handling mode LogOnly.\n\nDashboard can be filtered by namespaces as well as block types.", "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 0, @@ -88,7 +88,7 @@ "y": 0 }, "id": 6, - "title": "Airlock Microgateway Block Metrics", + "title": "Airlock Microgateway Threats LogOnly - Metrics", "type": "row" }, { @@ -96,81 +96,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Total number of requests processed by Airlock Microgateway.", - "fieldConfig": { - "defaults": { - "color": { - "fixedColor": "text", - "mode": "fixed" - }, - "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, - "unit": "short" - }, - "overrides": [] - }, - "gridPos": { - "h": 4, - "w": 4, - "x": 0, - "y": 1 - }, - "id": 1, - "options": { - "colorMode": "value", - "graphMode": "none", - "justifyMode": "auto", - "orientation": "auto", - "reduceOptions": { - "calcs": [ - "lastNotNull" - ], - "fields": "", - "values": false - }, - "showPercentChange": false, - "textMode": "auto", - "wideLayout": true - }, - "pluginVersion": "11.0.0", - "targets": [ - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "disableTextWrap": false, - "editorMode": "code", - "exemplar": false, - "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", - "format": "time_series", - "fullMetaSearch": false, - "hide": false, - "includeNullMetadata": true, - "instant": true, - "legendFormat": "Processed Requests", - "range": false, - "refId": "A", - "useBackend": false - } - ], - "title": "Requests", - "type": "stat" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", + "description": "Number of threats logged by Airlock Microgateway in threat handling mode LogOnly.", "fieldConfig": { "defaults": { "color": { @@ -198,14 +124,14 @@ } ] }, - "unit": "percentunit" + "unit": "short" }, "overrides": [] }, "gridPos": { "h": 4, "w": 4, - "x": 4, + "x": 0, "y": 1 }, "id": 2, @@ -214,6 +140,7 @@ "graphMode": "area", "justifyMode": "auto", "orientation": "auto", + "percentChangeColorMode": "standard", "reduceOptions": { "calcs": [ "last" @@ -225,7 +152,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -235,17 +162,17 @@ "disableTextWrap": false, "editorMode": "code", "exemplar": false, - "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", + "expr": "round(sum(increase(microgateway_http_downstream_rq_threats_logged_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", "fullMetaSearch": false, "includeNullMetadata": true, "instant": true, - "legendFormat": "Blocked Requests (%)", + "legendFormat": "Logged threats in LogOnly mode", "range": false, "refId": "A", "useBackend": false } ], - "title": "% Blocked Requests", + "title": "Threats - LogOnly", "type": "stat" }, { @@ -253,11 +180,11 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", + "description": "Number of threats per second handled in LogOnly mode.", "fieldConfig": { "defaults": { "color": { - "fixedColor": "blue", + "fixedColor": "orange", "mode": "fixed" }, "custom": { @@ -268,7 +195,7 @@ "axisPlacement": "left", "barAlignment": 0, "drawStyle": "line", - "fillOpacity": 0, + "fillOpacity": 25, "gradientMode": "none", "hideFrom": { "legend": false, @@ -301,53 +228,10 @@ "value": null } ] - } - }, - "overrides": [ - { - "matcher": { - "id": "byName", - "options": "% Blocks" - }, - "properties": [ - { - "id": "custom.axisPlacement", - "value": "right" - }, - { - "id": "unit", - "value": "percentunit" - }, - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - }, - { - "id": "max", - "value": 1 - } - ] }, - { - "matcher": { - "id": "byName", - "options": "Requests per second" - }, - "properties": [ - { - "id": "unit", - "value": "short" - }, - { - "id": "custom.fillOpacity", - "value": 25 - } - ] - } - ] + "unit": "short" + }, + "overrides": [] }, "gridPos": { "h": 10, @@ -380,27 +264,14 @@ }, "editorMode": "code", "exemplar": false, - "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "expr": "sum(rate(microgateway_http_downstream_rq_threats_logged_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", "instant": false, - "legendFormat": "Requests per second", + "legendFormat": "Number of threats per second", "range": true, - "refId": "Requests per Second" - }, - { - "datasource": { - "type": "prometheus", - "uid": "${DS_PROMETHEUS}" - }, - "editorMode": "code", - "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", - "hide": false, - "instant": false, - "legendFormat": "% Blocks", - "range": true, - "refId": "Blocks" + "refId": "LogOnly Events" } ], - "title": "Requests vs. % Blocks", + "title": "Threats - LogOnly", "type": "timeseries" }, { @@ -408,7 +279,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Blocked requests by block type.", + "description": "Number of threats in LogOnly mode by block type.", "fieldConfig": { "defaults": { "color": { @@ -448,7 +319,7 @@ } ] }, - "unit": "none" + "unit": "short" }, "overrides": [] }, @@ -482,7 +353,7 @@ "xTickLabelRotation": 0, "xTickLabelSpacing": 0 }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -491,7 +362,7 @@ }, "editorMode": "code", "exemplar": false, - "expr": "round(sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "expr": "round(sum by (block_type) (increase(microgateway_http_downstream_rq_threats_logged_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", "format": "time_series", "instant": true, "legendFormat": "__auto", @@ -520,7 +391,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Blocked requests by attack type, which are subsets of the various block types.", + "description": "Number of threats in LogOnly mode by block subtype, which are subsets of the various block types.", "fieldConfig": { "defaults": { "color": { @@ -557,7 +428,8 @@ "value": null } ] - } + }, + "unit": "short" }, "overrides": [] }, @@ -587,11 +459,11 @@ "mode": "single", "sort": "none" }, - "xField": "attack_type", + "xField": "block_subtype", "xTickLabelRotation": 0, "xTickLabelSpacing": 0 }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -600,14 +472,14 @@ }, "editorMode": "code", "exemplar": false, - "expr": "round(sum by (attack_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "expr": "round(sum by (block_subtype) (increase(microgateway_http_downstream_rq_threats_logged_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", "instant": true, "legendFormat": "__auto", "range": false, "refId": "A" } ], - "title": "Attack Type", + "title": "Block Subtype", "transformations": [ { "id": "reduce", @@ -630,11 +502,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "Datasource Prometheus", @@ -648,11 +516,7 @@ "type": "datasource" }, { - "current": { - "selected": false, - "text": "Loki", - "value": "P8E80F9AEF21F6940" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_LOKI", @@ -722,7 +586,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "definition": "label_values(microgateway_http_downstream_rq_threats_logged_total,block_type)", "hide": 0, "includeAll": true, "label": "Block Type", @@ -731,7 +595,7 @@ "options": [], "query": { "qryType": 1, - "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "query": "label_values(microgateway_http_downstream_rq_threats_logged_total,block_type)", "refId": "PrometheusVariableQueryEditor-VariableQuery" }, "refresh": 2, @@ -746,13 +610,12 @@ "from": "now-24h", "to": "now" }, - "timeRangeUpdatedDuringEditOrView": false, "timepicker": { "hidden": false }, "timezone": "browser", - "title": "Airlock Microgateway Block Metrics", - "uid": "ddnqoczu7qvb4cdd3dd", - "version": 3, + "title": "Airlock Microgateway Threats LogOnly - Metrics", + "uid": "ddnqoczu7qv2mfmsd3dd", + "version": 1, "weekStart": "" } \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.3/dashboards/overview.json b/charts/airlock/microgateway/4.4.1/dashboards/overview.json similarity index 98% rename from charts/airlock/microgateway/4.3.3/dashboards/overview.json rename to charts/airlock/microgateway/4.4.1/dashboards/overview.json index 094276621..8a9c913b0 100644 --- a/charts/airlock/microgateway/4.3.3/dashboards/overview.json +++ b/charts/airlock/microgateway/4.4.1/dashboards/overview.json @@ -137,7 +137,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -205,7 +205,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -290,7 +290,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -376,7 +376,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -566,7 +566,7 @@ "type": "prometheus", "uid": "${DS_PROMETHEUS}" }, - "description": "Requests blocked by Airlock Microgateway categorized by their corresponding type.", + "description": "Threats blocked by Airlock Microgateway categorized by their corresponding block type.", "fieldConfig": { "defaults": { "color": { @@ -670,7 +670,7 @@ } ] }, - "pluginVersion": "11.0.0", + "pluginVersion": "10.2.0", "targets": [ { "datasource": { @@ -686,7 +686,7 @@ "refId": "Block Types" } ], - "title": "Blocked Requests by Type", + "title": "Blocked Threats by Block Type", "transformations": [ { "id": "timeSeriesTable", @@ -1055,11 +1055,7 @@ "templating": { "list": [ { - "current": { - "selected": false, - "text": "Prometheus", - "value": "PBFA97CFB590B2093" - }, + "current": {}, "hide": 2, "includeAll": false, "label": "DS_PROMETHEUS", diff --git a/charts/airlock/microgateway/4.4.1/templates/NOTES.txt b/charts/airlock/microgateway/4.4.1/templates/NOTES.txt new file mode 100644 index 000000000..a607483f9 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/templates/NOTES.txt @@ -0,0 +1,61 @@ +Thank you for installing Airlock Microgateway. +{{- if .Values.operator.gatewayAPI.enabled }} + +K8s Gateway API support enabled. +Note that the K8s Gateway API support is an incubating Airlock Microgateway feature. We encourage you to try the installation and configuration for testing and evaluation. Your feedback is welcome. + + {{- if or .Values.operator.watchNamespaces .Values.operator.watchNamespaceSelector -}} + {{- fail ` + +K8s Gateway API is only supported using the 'AllNamespaces' installation mode type, ensure that 'operator.watchNamespaces' and 'operator.watchNamespaceSelector' are not configured. +` + -}} + {{- end -}} +{{- end }} + +Please ensure the following prerequisites are fulfilled: +* cert-manager is installed. + https://cert-manager.io/docs/installation/helm/ +* A valid Airlock Microgateway license is deployed in the Kubernetes secret '{{ .Release.Namespace }}/{{ .Values.license.secretName }}' + * Get a free Community license: https://airlock.com/en/microgateway-community + * Order a Premium license: https://airlock.com/en/microgateway-premium +* Airlock Microgateway CNI is installed on the cluster, when running data plane mode sidecar + https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni. + For more information about data plane modes, see https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }}/#data/1660804709650.html + +Further information: +* Documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }} +* CRD API reference documentation: https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" . }}/api/crds +* Airlock Microgateway Labs: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=helm +{{- if .Values.crds.skipVersionCheck }} + +Warning: CRD version check skipped +{{- else -}} +{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} +{{- if $outdatedCRDs -}} + {{- fail (printf ` + +Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. +Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: + +kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts + +If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` + .Chart.AppVersion) + -}} +{{- end -}} +{{- end -}} +{{- if .Values.tests.enabled -}} + {{- if .Values.operator.watchNamespaces -}} + {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) -}} + {{- fail (printf ` + +To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. +` + .Release.Namespace) + -}} + {{- end -}} + {{- end -}} +{{- end }} + +Your release version is {{ .Chart.Version }}. \ No newline at end of file diff --git a/charts/airlock/microgateway/4.4.1/templates/_helpers.tpl b/charts/airlock/microgateway/4.4.1/templates/_helpers.tpl new file mode 100644 index 000000000..733ba9648 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/templates/_helpers.tpl @@ -0,0 +1,153 @@ +{{/* +Expand the name of the chart. +We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest explicit suffix is 14 characters. +*/}} +{{- define "airlock-microgateway.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- end }} + +{{/* +Convert an image configuration object into an image ref string. +*/}} +{{- define "airlock-microgateway.image" -}} + {{- if .digest -}} + {{- printf "%s@%s" .repository .digest -}} + {{- else if .tag -}} + {{- printf "%s:%s" .repository .tag -}} + {{- else -}} + {{- printf "%s" .repository -}} + {{- end -}} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest implicit suffix is 27 characters. +If release name contains chart name it will be used as a full name. +*/}} +{{- define "airlock-microgateway.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "airlock-microgateway.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "airlock-microgateway.sharedLabels" -}} +helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/part-of: {{ .Chart.Name }} +{{- with .Values.commonLabels }} +{{ toYaml .}} +{{- end }} +{{- end }} + +{{/* +Common Selector labels +*/}} +{{- define "airlock-microgateway.sharedSelectorLabels" -}} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Restricted Container Security Context +*/}} +{{- define "airlock-microgateway.restrictedSecurityContext" -}} +allowPrivilegeEscalation: false +privileged: false +runAsNonRoot: true +capabilities: + drop: ["ALL"] +readOnlyRootFilesystem: true +seccompProfile: + type: RuntimeDefault +{{- end }} + +{{/* Precondition: May only be used if AppVersion is isSemver */}} +{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} +{{- $version := (semver .Chart.AppVersion) -}} +{{- if $version.Prerelease -}} +>= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} +{{- else -}} +>= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.outdatedCRDs" -}} +{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} + {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} + {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} + {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} + {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} + {{- $isOutdated := false -}} + {{- if $crd -}} + {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} + {{- $isOutdated = true -}} + {{- if hasKey $crd.metadata "labels" -}} + {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} + {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} + {{- if (semverCompare $supportedVersion $crdVersion) }} + {{- $isOutdated = false -}} + {{- end }} + {{- end -}} + {{- end -}} + {{- end -}} + {{- if $isOutdated }} +{{ base $path }} + {{- end }} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.isSemver" -}} +{{- regexMatch `^(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-]+)*))?$` . -}} +{{- end -}} + +{{- define "airlock-microgateway.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} + {{- $version := (semver .Chart.AppVersion) -}} + {{- $version.Major }}.{{ $version.Minor -}} +{{- else -}} + {{- print "latest" -}} +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} +{{- $list := list -}} +{{- with .matchLabels -}} + {{- range $key, $value := . -}} + {{- $list = append $list (printf "%s=%s" $key $value) -}} + {{- end -}} +{{- end -}} +{{- with .matchExpressions -}} + {{- range . -}} + {{- if has .operator (list "In" "NotIn") -}} + {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} + {{- else if eq .operator "Exists" -}} + {{- $list = append $list .key -}} + {{- else if eq .operator "DoesNotExist" -}} + {{- $list = append $list (printf "!%s" .key) -}} + {{- end -}} + {{- end -}} +{{- end -}} +{{- join "," $list -}} +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.4.1/templates/operator/_operator_helpers.tpl similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/_operator_helpers.tpl rename to charts/airlock/microgateway/4.4.1/templates/operator/_operator_helpers.tpl diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.4.1/templates/operator/_rbac.gen.tpl similarity index 73% rename from charts/airlock/microgateway/4.2.3/templates/operator/_rbac.gen.tpl rename to charts/airlock/microgateway/4.4.1/templates/operator/_rbac.gen.tpl index 528f72bc6..faa078b6b 100644 --- a/charts/airlock/microgateway/4.2.3/templates/operator/_rbac.gen.tpl +++ b/charts/airlock/microgateway/4.4.1/templates/operator/_rbac.gen.tpl @@ -8,6 +8,8 @@ Operator rbac permission rules - "" resources: - configmaps + - namespaces + - replicasets verbs: - get - list @@ -52,28 +54,107 @@ Operator rbac permission rules - delete - get - list + - patch - update - watch +- apiGroups: + - "" + resources: + - services + verbs: + - create + - get + - list + - patch + - update + - watch +- apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - get + - list + - watch +- apiGroups: + - apps + resources: + - deployments + verbs: + - create + - get + - list + - patch + - update + - watch +- apiGroups: + - apps + resources: + - replicasets + verbs: + - get + - list + - patch + - update + - watch +- apiGroups: + - apps + resources: + - replicasets/finalizers + verbs: + - patch + - update +- apiGroups: + - gateway.networking.k8s.io + resources: + - gatewayclasses + verbs: + - get + - list + - patch + - watch +- apiGroups: + - gateway.networking.k8s.io + resources: + - gatewayclasses/finalizers + - gatewayclasses/status + - gateways/finalizers + - gateways/status + - httproutes/status + verbs: + - patch + - update +- apiGroups: + - gateway.networking.k8s.io + resources: + - gateways + - httproutes + - referencegrants + verbs: + - get + - list + - watch - apiGroups: - microgateway.airlock.com resources: - accesscontrols - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - contentsecurities - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: + - contentsecuritypolicies - denyrules + - envoyclusters + - envoyhttpfilters + - graphqls + - headerrewrites + - identitypropagations + - jwks + - limits + - oidcproviders + - oidcrelyingparties + - openapis + - parsers + - redisproviders + - sessionhandlings + - telemetries verbs: - get - list @@ -81,11 +162,10 @@ Operator rbac permission rules - apiGroups: - microgateway.airlock.com resources: - - envoyclusters + - contentsecuritypolicies/status verbs: - - get - - list - - watch + - patch + - update - apiGroups: - microgateway.airlock.com resources: @@ -102,74 +182,11 @@ Operator rbac permission rules - microgateway.airlock.com resources: - envoyconfigurations/status + - sidecargateways/status verbs: - get - patch - update -- apiGroups: - - microgateway.airlock.com - resources: - - envoyhttpfilters - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - headerrewrites - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - identitypropagations - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - limits - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcproviders - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - oidcrelyingparties - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - openapis - verbs: - - get - - list - - watch -- apiGroups: - - microgateway.airlock.com - resources: - - parsers - verbs: - - get - - list - - watch - apiGroups: - microgateway.airlock.com resources: @@ -186,20 +203,4 @@ Operator rbac permission rules - sidecargateways/finalizers verbs: - update -- apiGroups: - - microgateway.airlock.com - resources: - - sidecargateways/status - verbs: - - get - - patch - - update -- apiGroups: - - microgateway.airlock.com - resources: - - telemetries - verbs: - - get - - list - - watch {{- end }} diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.4.1/templates/operator/_webhooks.gen.tpl similarity index 84% rename from charts/airlock/microgateway/4.3.1/templates/operator/_webhooks.gen.tpl rename to charts/airlock/microgateway/4.4.1/templates/operator/_webhooks.gen.tpl index 02e304890..97474df39 100644 --- a/charts/airlock/microgateway/4.3.1/templates/operator/_webhooks.gen.tpl +++ b/charts/airlock/microgateway/4.4.1/templates/operator/_webhooks.gen.tpl @@ -76,6 +76,26 @@ Operator validating webhooks resources: - accesscontrols sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-contentsecuritypolicy + failurePolicy: Fail + name: validate-contentsecuritypolicy.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - contentsecuritypolicies + sideEffects: None - admissionReviewVersions: - v1 clientConfig: @@ -196,6 +216,26 @@ Operator validating webhooks resources: - identitypropagations sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-jwks + failurePolicy: Fail + name: validate-jwks.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - jwks + sideEffects: None - admissionReviewVersions: - v1 clientConfig: @@ -316,6 +356,26 @@ Operator validating webhooks resources: - redisproviders sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-sessionhandling + failurePolicy: Fail + name: validate-sessionhandling.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - sessionhandlings + sideEffects: None - admissionReviewVersions: - v1 clientConfig: diff --git a/charts/airlock/microgateway/4.3.1/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/configmap.yaml similarity index 97% rename from charts/airlock/microgateway/4.3.1/templates/operator/configmap.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/configmap.yaml index 95e52d7df..276a632e8 100644 --- a/charts/airlock/microgateway/4.3.1/templates/operator/configmap.yaml +++ b/charts/airlock/microgateway/4.4.1/templates/operator/configmap.yaml @@ -190,8 +190,8 @@ data: stats_tags: - tag_name: "block_type" regex: "\\.(block_type\\.([^.]+))" - - tag_name: "attack_type" - regex: "\\.(attack_type\\.([^.]+))" + - tag_name: "block_subtype" + regex: "\\.(block_subtype\\.([^.]+))" - tag_name: "envoy_cluster_name" regex: "\\.(cluster\\.([^.]+))" - tag_name: "version" @@ -364,6 +364,10 @@ data: securityContext: {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} runAsUser: $(SECURITYCONTEXT_UID) + {{- with .Values.networkValidator.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} operator_config.yaml: | apiVersion: config.airlock.com/v1alpha1 kind: OperatorConfig @@ -392,3 +396,10 @@ data: list: {{- toYaml . | nindent 8 }} {{- end }} + {{- with $.Values.operator.gatewayAPI }} + gatewayAPI: + enabled: {{ .enabled }} + {{- if .controllerName }} + controllerName: {{ .controllerName }} + {{- end }} + {{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/dashboard-configmap.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/dashboard-configmap.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/deployment.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/deployment.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/manager-role.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/manager-role.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/manager-rolebinding.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/manager-rolebinding.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/metrics-service.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/metrics-service.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/metrics-service.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/mutating-webhook.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/mutating-webhook.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/podmonitor.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/podmonitor.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/role.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/role.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/role.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/role.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/rolebinding.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/rolebinding.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/rolebinding.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/selfsigned-issuer.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/selfsigned-issuer.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/selfsigned-issuer.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/serviceaccount.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/serviceaccount.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/serviceaccount.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/servicemonitor.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/servicemonitor.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/servicemonitor.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/serving-certificate.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/serving-certificate.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/serving-certificate.yaml diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/validating-webhook.yaml similarity index 100% rename from charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/validating-webhook.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/webhook-service.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/webhook-service.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/webhook-service.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.4.1/templates/operator/xds-service.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/operator/xds-service.yaml rename to charts/airlock/microgateway/4.4.1/templates/operator/xds-service.yaml diff --git a/charts/airlock/microgateway/4.4.1/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.4.1/templates/tests/rbac.yaml new file mode 100644 index 000000000..93bd4cd1b --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/templates/tests/rbac.yaml @@ -0,0 +1,143 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: "{{ include "airlock-microgateway.fullname" . }}-tests" +subjects: +- kind: ServiceAccount + name: "{{ include "airlock-microgateway.fullname" . }}-tests" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + verbs: + - get + - list + - watch + - delete +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways + verbs: + - create +- apiGroups: + - "" + resources: + - events + verbs: + - list +- apiGroups: + - "apps" + resources: + - deployments + resourceNames: + - "{{ include "airlock-microgateway.operator.fullname" . }}" + verbs: + - get + - list + - watch +- apiGroups: + - "apps" + resources: + - statefulsets + - statefulsets/scale + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + verbs: + - get + - list + - watch + - patch +- apiGroups: + - "" + resources: + - pods + - pods/log + - pods/status + - pods/attach + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" + - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" + - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" + verbs: + - get + - list + - create + - watch + - delete +- apiGroups: + - "" + resources: + - pods + verbs: + - create +{{- if .Values.operator.watchNamespaceSelector }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +subjects: + - kind: ServiceAccount + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +rules: +- apiGroups: + - "" + resources: + - namespaces + verbs: + - get + - list +{{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.2.3/templates/tests/service.yaml b/charts/airlock/microgateway/4.4.1/templates/tests/service.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/tests/service.yaml rename to charts/airlock/microgateway/4.4.1/templates/tests/service.yaml diff --git a/charts/airlock/microgateway/4.2.3/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.4.1/templates/tests/statefulset.yaml similarity index 100% rename from charts/airlock/microgateway/4.2.3/templates/tests/statefulset.yaml rename to charts/airlock/microgateway/4.4.1/templates/tests/statefulset.yaml diff --git a/charts/airlock/microgateway/4.4.1/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.4.1/templates/tests/test-install.yaml new file mode 100644 index 000000000..721ae2b82 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/templates/tests/test-install.yaml @@ -0,0 +1,227 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + sidecar.istio.io/inject: "false" + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + restartPolicy: Never + containers: + - name: test + image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + command: + - sh + - -c + - | + set -eu + + clean_up() { + echo "" + echo "### Clean up test resources" + kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true + echo "" + echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s + sleep 3s + echo "" + } + + fail() { + echo "" + echo "### Error: ${1}" + echo "" + + if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then + echo "" + echo 'Microgateway Sidecargateway status:' + kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true + echo "" + echo "" + fi + + if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then + echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" + kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true + echo "" + echo "" + echo 'Logs of Nginx container:' + kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true + echo "" + echo "" + # Wait for engine logs + sleep 10s + echo 'Logs of Microgateway Engine container:' + kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true + fi + + exit 1 + } + + create_sidecargateway() { + # create SidecarGateway resource for testing purposes + kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true + kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done + kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request + kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request + } + + {{- if .Values.operator.watchNamespaceSelector }} + echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" + if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then + labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') + fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` + .Release.Namespace + (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) + }} + fi + echo "" + {{- end }} + + trap clean_up EXIT + echo "" + + echo "### Waiting for Microgateway Operator Deployments to be ready" + if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ + deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then + fail 'Timeout occurred' + fi + echo "" + + echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" + # scale to zero replicas to ensure no pods are present from previous runs + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s + echo "" + + echo "### Waiting for backend pod" + i=0 + while true; do + if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then + break + elif [ $i -gt 3 ]; then + fail 'Pod not ready' + fi + sleep 2s + i=$((i+1)) + done + + echo "### Checking Microgateway Engine sidecar container was injected" + if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then + fail 'Microgateway Engine sidecar container not injected' + fi + echo "True" + echo "" + + echo "### Checking for valid license" + i=0 + while true; do + if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then + break + elif [ $i -gt 30 ]; then + fail 'Microgateway license is missing or invalid' + fi + sleep 2s + i=$((i+1)) + done + echo "True" + echo "" + + echo "### Create SidecarGateway resource for testing" + if ! create_sidecargateway ; then + fail 'Creation of SidecarGateway resource failed' + fi + echo "" + + echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" + if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then + fail 'Timeout occurred' + fi + echo "" + + echo "### Waiting for 'engine-config-valid' condition" + if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then + fail 'Configuration was never accepted by the Microgateway Engine' + fi + sleep 5s + echo "" + echo "" + + echo "### Checking whether a valid request is successful and returns HTTP status code '200'" + out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) + echo "Response:" + echo "${out}" + if ! echo "${out}" | grep -q "200 OK"; then + fail 'A valid request was not successful' + fi + echo "" + echo "" + + echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" + out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) + echo "Response:" + echo "${out}" + if ! echo "${out}" | grep -q "400 Bad Request"; then + fail 'A malicious request was not blocked' + fi + echo "" + echo "" + + echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" + exit 0 + serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" +{{- end -}} diff --git a/charts/airlock/microgateway/4.4.1/values.schema.json b/charts/airlock/microgateway/4.4.1/values.schema.json new file mode 100644 index 000000000..05c7d7717 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/values.schema.json @@ -0,0 +1,572 @@ +{ + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "nameOverride": { + "type": "string" + }, + "fullnameOverride": { + "type": "string" + }, + "commonLabels": { + "$ref": "#/definitions/StringMap" + }, + "commonAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "crds": { + "type": "object", + "properties": { + "skipVersionCheck": { + "type": "boolean" + } + }, + "additionalProperties": false + }, + "imagePullSecrets": { + "type": "array", + "items": { + "type": "object", + "properties": { + "name": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "name" + ], + "additionalProperties": true + } + }, + "operator": { + "type": "object", + "properties": { + "replicaCount": { + "type": "integer", + "minimum": 0 + }, + "updateStrategy": { + "$ref": "#/definitions/UpdateStrategy" + }, + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "serviceAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "serviceLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "tolerations": { + "type": "array", + "items": { + "type": "object" + } + }, + "affinity": { + "type": "object" + }, + "config": { + "type": "object", + "properties": { + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + } + }, + "required": [ + "logLevel" + ], + "additionalProperties": false + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "watchNamespaces": { + "type": "array", + "items": { + "type": "string" + } + }, + "watchNamespaceSelector": { + "$ref": "#/definitions/LabelSelector" + }, + "rbac": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + }, + "serviceMonitor": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "labels": { + "$ref": "#/definitions/StringMap" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + }, + "gatewayAPI": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "controllerName" : { + "type": "string", + "pattern": "^microgateway\\.airlock\\.com\/[A-Za-z0-9\/\\-._~%!$&'()*+,;=:]+$" + } + }, + "required": [ + "enabled" + ], + "additionalProperties": false + } + }, + "oneOf": [ + { + "properties": { + "watchNamespaces": { + "minItems": 1 + }, + "watchNamespaceSelector": { + "additionalProperties": false + } + } + }, + { + "properties": { + "watchNamespaces": { + "maxItems": 0 + }, + "watchNamespaceSelector": { + "$ref": "#/definitions/LabelSelector" + } + } + } + ], + "required": [ + "affinity", + "config", + "image", + "updateStrategy", + "nodeSelector", + "podAnnotations", + "podLabels", + "rbac", + "replicaCount", + "resources", + "serviceAccount", + "serviceAnnotations", + "serviceLabels", + "serviceMonitor", + "tolerations" + ], + "additionalProperties": false + }, + "engine": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + }, + "resources": { + "type": "object" + }, + "sidecar": { + "type": "object", + "properties":{ + "podMonitor": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "labels": { + "$ref": "#/definitions/StringMap" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + } + }, + "required": [ + "podMonitor" + ], + "additionalProperties": false + } + }, + "required": [ + "image", + "resources", + "sidecar" + ], + "additionalProperties": false + }, + "networkValidator": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + }, + "resources": { + "type": "object" + } + }, + "required": [ + "image", + "resources" + ], + "additionalProperties": false + }, + "sessionAgent": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + }, + "resources": { + "type": "object" + } + }, + "required": [ + "image", + "resources" + ], + "additionalProperties": false + }, + "license": { + "type": "object", + "properties": { + "secretName": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "secretName" + ], + "additionalProperties": false + }, + "dashboards": { + "type": "object", + "properties" : { + "create": { + "type": "boolean" + }, + "config": { + "type": "object", + "properties": { + "grafana": { + "type": "object", + "properties": { + "folderAnnotation": { + "$ref": "#/definitions/NameValuePair" + }, + "dashboardLabel": { + "$ref": "#/definitions/NameValuePair" + } + }, + "required": [ + "folderAnnotation", + "dashboardLabel" + ], + "additionalProperties": false + } + }, + "required": [ + "grafana" + ], + "additionalProperties": false + }, + "instances": { + "type": "object", + "properties": { + "overview": { + "$ref": "#/definitions/DashboardInstance" + }, + "license" : { + "$ref": "#/definitions/DashboardInstance" + }, + "blockMetrics" : { + "$ref": "#/definitions/DashboardInstance" + }, + "blockLogs" : { + "$ref": "#/definitions/DashboardInstance" + }, + "headerLogs" : { + "$ref": "#/definitions/DashboardInstance" + }, + "logOnlyMetrics" : { + "$ref": "#/definitions/DashboardInstance" + }, + "logOnlyLogs" : { + "$ref": "#/definitions/DashboardInstance" + } + }, + "required": [ + "overview", + "license", + "blockMetrics", + "blockLogs", + "headerLogs", + "logOnlyMetrics", + "logOnlyLogs" + ], + "additionalProperties": false + } + }, + "required": [ + "create", + "config", + "instances" + ], + "additionalProperties": false + }, + "tests": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "additionalProperties": false + }, + "global": { + "type": "object" + } + }, + "required": [ + "commonAnnotations", + "commonLabels", + "crds", + "engine", + "fullnameOverride", + "imagePullSecrets", + "license", + "nameOverride", + "operator", + "networkValidator", + "sessionAgent", + "dashboards", + "tests" + ], + "additionalProperties": false, + "definitions": { + "StringMap": { + "type": "object", + "additionalProperties": { + "type": "string" + } + }, + "Image": { + "type": "object", + "properties": { + "repository": { + "type": "string", + "minLength": 1 + }, + "tag": { + "type": "string" + }, + "digest": { + "type": "string", + "pattern": "^$|^sha256:[a-f0-9]{64}$" + }, + "pullPolicy": { + "type": "string", + "enum": [ + "Always", + "IfNotPresent", + "Never" + ] + } + }, + "required": [ + "digest", + "pullPolicy", + "repository", + "tag" + ], + "additionalProperties": false + }, + "LabelSelector": { + "type": "object", + "properties": { + "matchExpressions": { + "type": "array", + "items": { + "type": "object", + "required": [ + "key", + "operator" + ], + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "additionalProperties": false + } + }, + "matchLabels": { + "$ref": "#/definitions/StringMap" + } + }, + "additionalProperties": false + }, + "UpdateStrategy": { + "type": "object", + "oneOf" : [ + { + "properties": { + "type": { + "$ref": "#/definitions/RecreateType" + } + }, + "required": [ + "type" + ], + "additionalProperties": false + }, + { + "properties": { + "type": { + "$ref": "#/definitions/RollingUpdateType" + }, + "rollingUpdate": { + "$ref": "#/definitions/RollingUpdate" + } + }, + "required": [ + "type" + ], + "additionalProperties": false + } + ] + }, + "RecreateType": { + "type": "string", + "enum": [ + "Recreate" + ] + }, + "RollingUpdateType": { + "type": "string", + "enum": [ + "RollingUpdate" + ] + }, + "RollingUpdate": { + "type": "object", + "properties": { + "maxSurge": { + "type": ["integer", "string"], + "minimum": 0, + "pattern": "^\\d+%?$" + }, + "maxUnavailable": { + "type": ["integer", "string"], + "minimum": 0, + "pattern": "^\\d+%?$" + } + }, + "anyOf": [ + {"required": ["maxSurge"]}, + {"required": ["maxUnavailable"]} + ], + "additionalProperties": false + }, + "DashboardInstance" : { + "type" : "object", + "properties" : { + "create" : { + "type" : "boolean" + } + }, + "required" : [ + "create" + ], + "additionalProperties": false + }, + "NameValuePair" : { + "type" : "object", + "properties" : { + "name" : { + "type": "string", + "minLength": 1 + }, + "value" : { + "type" : "string", + "minLength": 1 + } + }, + "required" : [ + "name", + "value" + ], + "additionalProperties": false + } + } +} diff --git a/charts/airlock/microgateway/4.4.1/values.yaml b/charts/airlock/microgateway/4.4.1/values.yaml new file mode 100644 index 000000000..f0f598ea1 --- /dev/null +++ b/charts/airlock/microgateway/4.4.1/values.yaml @@ -0,0 +1,237 @@ +# -- Allows overriding the name to use instead of "microgateway". +nameOverride: "" +# -- Allows overriding the name to use as full name of resources. +fullnameOverride: "" +# -- Labels to add to all resources. +commonLabels: {} +# -- Annotations to add to all resources. +commonAnnotations: {} +# -- ImagePullSecrets to use when pulling images. +imagePullSecrets: [] +# - name: myRegistryKeySecretName + +crds: + # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. + # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster + # when performing a "helm install/upgrade". + skipVersionCheck: false +operator: + # -- Number of replicas for the operator Deployment. + replicaCount: 2 + # -- Specifies the operator update strategy. + updateStrategy: + type: RollingUpdate + # Specifies the Airlock Microgateway Operator image. + image: + # -- Image repository from which to pull the Airlock Microgateway Operator image. + repository: "quay.io/airlock/microgateway-operator" + # -- Image tag to pull. + tag: "4.4.1" + # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). + # Overrides tag when specified. + digest: "sha256:1133c3e59418eec1721683e68dd19faca577609ace6eebd010a56e52b1f75789" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Annotations to add to all Pods. + podAnnotations: {} + # -- Labels to add to all Pods. + podLabels: {} + # -- Annotations to add to the Service. + serviceAnnotations: {} + # prometheus.io/scrape: "true" + # prometheus.io/port: "8080" + + # -- Labels to add to the Service. + serviceLabels: {} + # -- Resource restrictions to apply to the operator container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 1000m + # memory: 512Mi + # requests: + # cpu: 100m + # memory: 512Mi + + # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. + nodeSelector: {} + # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. + tolerations: [] + # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. + affinity: {} + # Parameters for the operator configuration. + config: + # -- Operator application log level. + logLevel: "info" + # Configures the generation of the ServiceAccount. + serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" + # -- Allows to restrict the operator to specific namespaces, depending on your needs. + # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). + # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. + # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. + # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. + # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. + # Please note that this feature requires a Premium license. + watchNamespaces: [] + # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. + # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. + # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). + # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. + # Please note that this feature requires a Premium license. + watchNamespaceSelector: {} + # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. + # matchLabels: + # microgateway.airlock.com/enable: "true" + # matchExpressions: + # - { key: environment, operator: NotIn, values: [dev] } + + # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. + rbac: + # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. + create: true + # Configures the generation of a Prometheus Operator ServiceMonitor. + serviceMonitor: + # -- Whether to create a ServiceMonitor resource for monitoring. + create: false + # -- Labels to add to the ServiceMonitor. + labels: {} + # release: "" + # Configures the Kubernetes Gateway API integration. + gatewayAPI: + # -- Whether to enable the Kubernetes Gateway API related controllers. + # Requires that the gateway.networking.k8s.io/v1 resources are installed on the cluster. + enabled: false + # -- Controller name referred in the GatewayClasses managed by this operator. The value must be a path prefixed by the domain `microgateway.airlock.com`. + controllerName: microgateway.airlock.com/gatewayclass-controller +engine: + # Specifies the Airlock Microgateway Engine image. + image: + # -- Image repository from which to pull the Airlock Microgateway Engine image. + repository: "quay.io/airlock/microgateway-engine" + # -- Image tag to pull. + tag: "4.4.1" + # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). + # Overrides tag when specified. + digest: "sha256:06573ef5e6769dbd6eb8606e34c56f1ad2084b6adcae9925b1d2d153a45cbc47" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Resource restrictions to apply to the Airlock Microgateway Engine container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 500m + # memory: 128Mi + # requests: + # cpu: 10m + # memory: 40Mi + + # Additional configuration when deployed as a sidecar. + sidecar: + # Configures the generation of a Prometheus Operator PodMonitor. + podMonitor: + # -- Whether to create a PodMonitor resource for monitoring. + create: false + # -- Labels to add to the PodMonitor. + labels: {} + # release: "" +networkValidator: + # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. + image: + # -- Image repository from which to pull the netcat image for the Airlock Microgateway Network Validator init-container. + repository: "cgr.dev/chainguard/netcat" + # -- Image tag to pull. + tag: "" + # -- SHA256 image digest to pull (in the format "sha256:7ef657ce316ce9d86f90c1dc99702d1190877c6ac2e923e696dc82c30050a14c"). + # Overrides tag when specified. + digest: "sha256:7ef657ce316ce9d86f90c1dc99702d1190877c6ac2e923e696dc82c30050a14c" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Resource restrictions to apply to the Airlock Microgateway Network Validator init-container. + resources: + limits: + cpu: 25m + memory: 12Mi + requests: + cpu: 5m + memory: 1Mi +sessionAgent: + # Specifies the Airlock Microgateway Session Agent image. + image: + # -- Image repository from which to pull the Airlock Microgateway Session Agent image. + repository: "quay.io/airlock/microgateway-session-agent" + # -- Image tag to pull. + tag: "4.4.1" + # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). + # Overrides tag when specified. + digest: "sha256:733a25f61ea7cf43c0a46da7d3ecb9a263bda49bf60e1fd8e4162be33aa24b7b" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 150m + # memory: 32Mi + # requests: + # cpu: 10m + # memory: 8Mi +license: + # -- Name of the secret containing the "microgateway-license.txt" key. + secretName: "airlock-microgateway-license" +# Creates dashboards in the form of ConfigMaps that can be imported +# by Grafana using its sidecar setup. +dashboards: + # -- Whether to create any ConfigMaps containing Grafana dashboards to import. + create: false + config: + # Configures the necessary label and annotations along with their values + # to enable Grafana to correctly identify the ConfigMaps containing + # dashboards and file them within a dedicated folder in the dashboard overview. + # These settings need to match the Grafana sidecar configuration. + grafana: + folderAnnotation: + # -- Name of the annotation containing the folder name to file dashboards into. + name: "grafana_folder" + # -- Name of the folder dashboards are filed into within the Grafana UI. + value: "Airlock Microgateway" + dashboardLabel: + # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. + name: "grafana_dashboard" + # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. + value: "1" + instances: + # Available dashboard instances that can be individually created/deployed. + overview: + # -- Whether to create the overview dashboard. + create: true + license: + # -- Whether to create the license dashboard. + create: true + blockMetrics: + # -- Whether to create the block metrics dashboard. + create: true + blockLogs: + # -- Whether to create the block logs dashboard. + create: true + headerLogs: + # -- Whether to create the header rewrite logs dashboard. + create: true + logOnlyMetrics: + # -- Whether to create the log only metrics dashboard + create: true + logOnlyLogs: + # -- Whether to create the log only logs dashboard. + create: true +# Check whether the installation of the Airlock Microgateway Helm Chart was successful. +# Requires a secret with a valid Airlock Microgateway license key already to be present. +tests: + # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). + # If set to false, `helm test` will not run any tests. + enabled: false diff --git a/charts/kasten/k10/7.0.1401/Chart.lock b/charts/kasten/k10/7.0.1401/Chart.lock new file mode 100644 index 000000000..b1b447114 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/Chart.lock @@ -0,0 +1,9 @@ +dependencies: +- name: grafana + repository: "" + version: 8.5.8 +- name: prometheus + repository: "" + version: 25.28.0 +digest: sha256:4af966da2fe3b7163bec3c5da98178c8232d6a6ad4405c8e592d38a6832ce704 +generated: "2024-11-15T16:25:38.370071605Z" diff --git a/charts/kasten/k10/7.0.1401/Chart.yaml b/charts/kasten/k10/7.0.1401/Chart.yaml new file mode 100644 index 000000000..3e1dd214a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/Chart.yaml @@ -0,0 +1,25 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: K10 + catalog.cattle.io/kube-version: '>= 1.17.0-0' + catalog.cattle.io/release-name: k10 +apiVersion: v2 +appVersion: 7.0.14 +dependencies: +- condition: grafana.enabled + name: grafana + repository: "" + version: 8.5.8 +- condition: prometheus.server.enabled + name: prometheus + repository: "" + version: 25.28.0 +description: Kasten’s K10 Data Management Platform +home: https://kasten.io/ +icon: file://assets/icons/k10.png +kubeVersion: '>= 1.17.0-0' +maintainers: +- email: contact@kasten.io + name: kastenIO +name: k10 +version: 7.0.1401 diff --git a/charts/kasten/k10/7.0.1401/README.md b/charts/kasten/k10/7.0.1401/README.md new file mode 100644 index 000000000..7dbccc9c7 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/README.md @@ -0,0 +1,344 @@ +# Kasten's K10 Helm chart. + +[Kasten's k10](https://docs.kasten.io/) is a data lifecycle management system for all your persistence.enabled +container-based applications. + +## TL;DR; + +```console +$ helm install kasten/k10 --name=k10 --namespace=kasten-io +``` +Additionally, K10 images are available in Platform One's **Iron Bank** hardened container registry. +To install using these images, follow the instructions found +[here](https://docs.kasten.io/latest/install/ironbank.html). + +## Introduction + +This chart bootstraps Kasten's K10 platform on a [Kubernetes](http://kubernetes.io) cluster using +the [Helm](https://helm.sh) package manager. + +## Prerequisites + +- Kubernetes 1.23 - 1.26 + +## Installing the Chart + +To install the chart on a [GKE](https://cloud.google.com/container-engine/) cluster + +```console +$ helm install kasten/k10 --name=k10 --namespace=kasten-io +``` + +To install the chart on an [AWS](https://aws.amazon.com/) [kops](https://github.com/kubernetes/kops)-created cluster + +```console +$ helm install kasten/k10 --name=k10 --namespace=kasten-io --set secrets.awsAccessKeyId="${AWS_ACCESS_KEY_ID}" \ + --set secrets.awsSecretAccessKey="${AWS_SECRET_ACCESS_KEY}" +``` + +> **Tip**: List all releases using `helm list` + +## Uninstalling the Chart + +To uninstall/delete the `k10` application: + +```console +$ helm delete k10 --purge +``` + +## Configuration + +The following table lists the configurable parameters of the K10 +chart and their default values. + +Parameter | Description | Default +--- | --- | --- +`eula.accept`| Whether to enable accept EULA before installation | `false` +`eula.company` | Company name. Required field if EULA is accepted | `None` +`eula.email` | Contact email. Required field if EULA is accepted | `None` +`license` | License string obtained from Kasten | `None` +`rbac.create` | Whether to enable RBAC with a specific cluster role and binding for K10 | `true` +`scc.create` | Whether to create a SecurityContextConstraints for K10 ServiceAccounts | `false` +`scc.priority` | Sets the SecurityContextConstraints priority | `15` +`services.dashboardbff.hostNetwork` | Whether the dashboardbff pods may use the node network | `false` +`services.executor.hostNetwork` | Whether the executor pods may use the node network | `false` +`services.aggregatedapis.hostNetwork` | Whether the aggregatedapis pods may use the node network | `false` +`serviceAccount.create`| Specifies whether a ServiceAccount should be created | `true` +`serviceAccount.name` | The name of the ServiceAccount to use. If not set, a name is derived using the release and chart names. | `None` +`ingress.create` | Specifies whether the K10 dashboard should be exposed via ingress | `false` +`ingress.name` | Optional name of the Ingress object for the K10 dashboard. If not set, the name is formed using the release name. | `{Release.Name}-ingress` +`ingress.class` | Cluster ingress controller class: `nginx`, `GCE` | `None` +`ingress.host` | FQDN (e.g., `k10.example.com`) for name-based virtual host | `None` +`ingress.urlPath` | URL path for K10 Dashboard (e.g., `/k10`) | `Release.Name` +`ingress.pathType` | Specifies the path type for the ingress resource | `ImplementationSpecific` +`ingress.annotations` | Additional Ingress object annotations | `{}` +`ingress.tls.enabled` | Configures a TLS use for `ingress.host` | `false` +`ingress.tls.secretName` | Optional TLS secret name | `None` +`ingress.defaultBackend.service.enabled` | Configures the default backend backed by a service for the K10 dashboard Ingress (mutually exclusive setting with `ingress.defaultBackend.resource.enabled`). | `false` +`ingress.defaultBackend.service.name` | The name of a service referenced by the default backend (required if the service-backed default backend is used). | `None` +`ingress.defaultBackend.service.port.name` | The port name of a service referenced by the default backend (mutually exclusive setting with port `number`, required if the service-backed default backend is used). | `None` +`ingress.defaultBackend.service.port.number` | The port number of a service referenced by the default backend (mutually exclusive setting with port `name`, required if the service-backed default backend is used). | `None` +`ingress.defaultBackend.resource.enabled` | Configures the default backend backed by a resource for the K10 dashboard Ingress (mutually exclusive setting with `ingress.defaultBackend.service.enabled`). | `false` +`ingress.defaultBackend.resource.apiGroup` | Optional API group of a resource backing the default backend. | `''` +`ingress.defaultBackend.resource.kind` | The type of a resource being referenced by the default backend (required if the resource default backend is used). | `None` +`ingress.defaultBackend.resource.name` | The name of a resource being referenced by the default backend (required if the resource default backend is used). | `None` +`global.persistence.size` | Default global size of volumes for K10 persistent services | `20Gi` +`global.persistence.catalog.size` | Size of a volume for catalog service | `global.persistence.size` +`global.persistence.jobs.size` | Size of a volume for jobs service | `global.persistence.size` +`global.persistence.logging.size` | Size of a volume for logging service | `global.persistence.size` +`global.persistence.metering.size` | Size of a volume for metering service | `global.persistence.size` +`global.persistence.storageClass` | Specified StorageClassName will be used for PVCs | `None` +`global.podLabels` | Configures custom labels to be set to all Kasten pods | `None` +`global.podAnnotations` | Configures custom annotations to be set to all Kasten pods | `None` +`global.airgapped.repository` | Specify the helm repository for offline (airgapped) installation | `''` +`global.imagePullSecret` | Provide secret which contains docker config for private repository. Use `k10-ecr` when secrets.dockerConfigPath is used. | `''` +`global.prometheus.external.host` | Provide external prometheus host name | `''` +`global.prometheus.external.port` | Provide external prometheus port number | `''` +`global.prometheus.external.baseURL` | Provide Base URL of external prometheus | `''` +`global.network.enable_ipv6` | Enable `IPv6` support for K10 | `false` +`google.workloadIdentityFederation.enabled` | Enable Google Workload Identity Federation for K10 | `false` +`google.workloadIdentityFederation.idp.type` | Identity Provider type for Google Workload Identity Federation for K10 | `''` +`google.workloadIdentityFederation.idp.aud` | Audience for whom the ID Token from Identity Provider is intended | `''` +`secrets.awsAccessKeyId` | AWS access key ID (required for AWS deployment) | `None` +`secrets.awsSecretAccessKey` | AWS access key secret | `None` +`secrets.awsIamRole` | ARN of the AWS IAM role assumed by K10 to perform any AWS operation. | `None` +`secrets.awsClientSecretName` | The secret that contains AWS access key ID, AWS access key secret and AWS IAM role for AWS | `None` +`secrets.googleApiKey` | Non-default base64 encoded GCP Service Account key | `None` +`secrets.googleProjectId` | Sets Google Project ID other than the one used in the GCP Service Account | `None` +`secrets.azureTenantId` | Azure tenant ID (required for Azure deployment) | `None` +`secrets.azureClientId` | Azure Service App ID | `None` +`secrets.azureClientSecret` | Azure Service APP secret | `None` +`secrets.azureClientSecretName` | The secret that contains ClientID, ClientSecret and TenantID for Azure | `None` +`secrets.azureResourceGroup` | Resource Group name that was created for the Kubernetes cluster | `None` +`secrets.azureSubscriptionID` | Subscription ID in your Azure tenant | `None` +`secrets.azureResourceMgrEndpoint` | Resource management endpoint for the Azure Stack instance | `None` +`secrets.azureADEndpoint` | Azure Active Directory login endpoint | `None` +`secrets.azureADResourceID` | Azure Active Directory resource ID to obtain AD tokens | `None` +`secrets.microsoftEntraIDEndpoint` | Microsoft Entra ID login endpoint | `None` +`secrets.microsoftEntraIDResourceID` | Microsoft Entra ID resource ID to obtain AD tokens | `None` +`secrets.azureCloudEnvID` | Azure Cloud Environment ID | `None` +`secrets.vsphereEndpoint` | vSphere endpoint for login | `None` +`secrets.vsphereUsername` | vSphere username for login | `None` +`secrets.vspherePassword` | vSphere password for login | `None` +`secrets.vsphereClientSecretName` | The secret that contains vSphere username, vSphere password and vSphere endpoint | `None` +`secrets.dockerConfig` | Set base64 encoded docker config to use for image pull operations. Alternative to the ``secrets.dockerConfigPath`` | `None` +`secrets.dockerConfigPath` | Use ``--set-file secrets.dockerConfigPath=path_to_docker_config.yaml`` to specify docker config for image pull. Will be overwritten if ``secrets.dockerConfig`` is set | `None` +`cacertconfigmap.name` | Name of the ConfigMap that contains a certificate for a trusted root certificate authority | `None` +`clusterName` | Cluster name for better logs visibility | `None` +`metering.awsRegion` | Sets AWS_REGION for metering service | `None` +`metering.mode` | Control license reporting (set to `airgap` for private-network installs) | `None` +`metering.reportCollectionPeriod` | Sets metric report collection period (in seconds) | `1800` +`metering.reportPushPeriod` | Sets metric report push period (in seconds) | `3600` +`metering.promoID` | Sets K10 promotion ID from marketing campaigns | `None` +`metering.awsMarketplace` | Sets AWS cloud metering license mode | `false` +`metering.awsManagedLicense` | Sets AWS managed license mode | `false` +`metering.redhatMarketplacePayg` | Sets Red Hat cloud metering license mode | `false` +`metering.licenseConfigSecretName` | Sets AWS managed license config secret | `None` +`externalGateway.create` | Configures an external gateway for K10 API services | `false` +`externalGateway.annotations` | Standard annotations for the services | `None` +`externalGateway.fqdn.name` | Domain name for the K10 API services | `None` +`externalGateway.fqdn.type` | Supported gateway type: `route53-mapper` or `external-dns` | `None` +`externalGateway.awsSSLCertARN` | ARN for the AWS ACM SSL certificate used in the K10 API server | `None` +`auth.basicAuth.enabled` | Configures basic authentication for the K10 dashboard | `false` +`auth.basicAuth.htpasswd` | A username and password pair separated by a colon character | `None` +`auth.basicAuth.secretName` | Name of an existing Secret that contains a file generated with htpasswd | `None` +`auth.k10AdminGroups` | A list of groups whose members are granted admin level access to K10's dashboard | `None` +`auth.k10AdminUsers` | A list of users who are granted admin level access to K10's dashboard | `None` +`auth.tokenAuth.enabled` | Configures token based authentication for the K10 dashboard | `false` +`auth.oidcAuth.enabled` | Configures Open ID Connect based authentication for the K10 dashboard | `false` +`auth.oidcAuth.providerURL` | URL for the OIDC Provider | `None` +`auth.oidcAuth.redirectURL` | URL to the K10 gateway service | `None` +`auth.oidcAuth.scopes` | Space separated OIDC scopes required for userinfo. Example: "profile email" | `None` +`auth.oidcAuth.prompt` | The type of prompt to be used during authentication (none, consent, login or select_account) | `select_account` +`auth.oidcAuth.clientID` | Client ID given by the OIDC provider for K10 | `None` +`auth.oidcAuth.clientSecret` | Client secret given by the OIDC provider for K10 | `None` +`auth.oidcAuth.clientSecretName` | The secret that contains the Client ID and Client secret given by the OIDC provider for K10 | `None` +`auth.oidcAuth.usernameClaim` | The claim to be used as the username | `sub` +`auth.oidcAuth.usernamePrefix` | Prefix that has to be used with the username obtained from the username claim | `None` +`auth.oidcAuth.groupClaim` | Name of a custom OpenID Connect claim for specifying user groups | `None` +`auth.oidcAuth.groupPrefix` | All groups will be prefixed with this value to prevent conflicts | `None` +`auth.oidcAuth.sessionDuration` | Maximum OIDC session duration | `1h` +`auth.oidcAuth.refreshTokenSupport` | Enable OIDC Refresh Token support | `false` +`auth.openshift.enabled` | Enables access to the K10 dashboard by authenticating with the OpenShift OAuth server | `false` +`auth.openshift.serviceAccount` | Name of the service account that represents an OAuth client | `None` +`auth.openshift.clientSecret` | The token corresponding to the service account | `None` +`auth.openshift.clientSecretName` | The secret that contains the token corresponding to the service account | `None` +`auth.openshift.dashboardURL` | The URL used for accessing K10's dashboard | `None` +`auth.openshift.openshiftURL` | The URL for accessing OpenShift's API server | `None` +`auth.openshift.insecureCA` | To turn off SSL verification of connections to OpenShift | `false` +`auth.openshift.useServiceAccountCA` | Set this to true to use the CA certificate corresponding to the Service Account ``auth.openshift.serviceAccount`` usually found at ``/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`` | `false` +`auth.openshift.caCertsAutoExtraction` | Set this to false to disable the OCP CA certificates automatic extraction to the K10 namespace | `true` +`auth.ldap.enabled` | Configures Active Directory/LDAP based authentication for the K10 dashboard | `false` +`auth.ldap.restartPod` | To force a restart of the authentication service pod (useful when updating authentication config) | `false` +`auth.ldap.dashboardURL` | The URL used for accessing K10's dashboard | `None` +`auth.ldap.host` | Host and optional port of the AD/LDAP server in the form `host:port` | `None` +`auth.ldap.insecureNoSSL` | Required if the AD/LDAP host is not using TLS | `false` +`auth.ldap.insecureSkipVerifySSL` | To turn off SSL verification of connections to the AD/LDAP host | `false` +`auth.ldap.startTLS` | When set to true, ldap:// is used to connect to the server followed by creation of a TLS session. When set to false, ldaps:// is used. | `false` +`auth.ldap.bindDN` | The Distinguished Name(username) used for connecting to the AD/LDAP host | `None` +`auth.ldap.bindPW` | The password corresponding to the `bindDN` for connecting to the AD/LDAP host | `None` +`auth.ldap.bindPWSecretName` | The name of the secret that contains the password corresponding to the `bindDN` for connecting to the AD/LDAP host | `None` +`auth.ldap.userSearch.baseDN` | The base Distinguished Name to start the AD/LDAP search from | `None` +`auth.ldap.userSearch.filter` | Optional filter to apply when searching the directory | `None` +`auth.ldap.userSearch.username` | Attribute used for comparing user entries when searching the directory | `None` +`auth.ldap.userSearch.idAttr` | AD/LDAP attribute in a user's entry that should map to the user ID field in a token | `None` +`auth.ldap.userSearch.emailAttr` | AD/LDAP attribute in a user's entry that should map to the email field in a token | `None` +`auth.ldap.userSearch.nameAttr` | AD/LDAP attribute in a user's entry that should map to the name field in a token | `None` +`auth.ldap.userSearch.preferredUsernameAttr` | AD/LDAP attribute in a user's entry that should map to the preferred_username field in a token | `None` +`auth.ldap.groupSearch.baseDN` | The base Distinguished Name to start the AD/LDAP group search from | `None` +`auth.ldap.groupSearch.filter` | Optional filter to apply when searching the directory for groups | `None` +`auth.ldap.groupSearch.nameAttr` | The AD/LDAP attribute that represents a group's name in the directory | `None` +`auth.ldap.groupSearch.userMatchers` | List of field pairs that are used to match a user to a group. | `None` +`auth.ldap.groupSearch.userMatchers.userAttr` | Attribute in the user's entry that must match with the `groupAttr` while searching for groups | `None` +`auth.ldap.groupSearch.userMatchers.groupAttr` | Attribute in the group's entry that must match with the `userAttr` while searching for groups | `None` +`auth.groupAllowList` | A list of groups whose members are allowed access to K10's dashboard | `None` +`services.securityContext` | Custom [security context](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/) for K10 service containers | `{"runAsUser" : 1000, "fsGroup": 1000}` +`services.securityContext.runAsUser` | User ID K10 service containers run as| `1000` +`services.securityContext.runAsGroup` | Group ID K10 service containers run as| `1000` +`services.securityContext.fsGroup` | FSGroup that owns K10 service container volumes | `1000` +`siem.logging.cluster.enabled` | Whether to enable writing K10 audit event logs to stdout (standard output) | `true` +`siem.logging.cloud.path` | Directory path for saving audit logs in a cloud object store | `k10audit/` +`siem.logging.cloud.awsS3.enabled` | Whether to enable sending K10 audit event logs to AWS S3 | `true` +`injectKanisterSidecar.enabled` | Enable Kanister sidecar injection for workload pods | `false` +`injectKanisterSidecar.namespaceSelector.matchLabels` | Set of labels to select namespaces in which sidecar injection is enabled for workloads | `{}` +`injectKanisterSidecar.objectSelector.matchLabels` | Set of labels to filter workload objects in which the sidecar is injected | `{}` +`injectKanisterSidecar.webhookServer.port` | Port number on which the mutating webhook server accepts request | `8080` +`gateway.insecureDisableSSLVerify` | Specifies whether to disable SSL verification for gateway pods | `false` +`gateway.exposeAdminPort` | Specifies whether to expose Admin port for gateway service | `true` +`gateway.resources.[requests\|limits].[cpu\|memory]` | Resource requests and limits for gateway pod | `{}` +`gateway.service.externalPort` | Specifies the gateway services external port | `80` +`genericVolumeSnapshot.resources.[requests\|limits].[cpu\|memory]` | Specifies resource requests and limits for generic backup sidecar and all temporary Kasten worker Pods. Superseded by ActionPodSpec | `{}` +`multicluster.enabled` | Choose whether to enable the multi-cluster system components and capabilities | `true` +`multicluster.primary.create` | Choose whether to setup cluster as a multi-cluster primary | `false` +`multicluster.primary.name` | Primary cluster name | `''` +`multicluster.primary.ingressURL` | Primary cluster dashboard URL | `''` +`prometheus.k10image.registry` | (optional) Set Prometheus image registry. | `gcr.io` +`prometheus.k10image.repository` | (optional) Set Prometheus image repository. | `kasten-images` +`prometheus.rbac.create` | (optional) Whether to create Prometheus RBAC configuration. Warning - this action will allow prometheus to scrape pods in all k8s namespaces | `false` +`prometheus.alertmanager.enabled` | DEPRECATED: (optional) Enable Prometheus `alertmanager` service | `false` +`prometheus.alertmanager.serviceAccount.create` | DEPRECATED: (optional) Set true to create ServiceAccount for `alertmanager` | `false` +`prometheus.networkPolicy.enabled` | DEPRECATED: (optional) Enable Prometheus `networkPolicy` | `false` +`prometheus.prometheus-node-exporter.enabled` | DEPRECATED: (optional) Enable Prometheus `node-exporter` | `false` +`prometheus.prometheus-node-exporter.serviceAccount.create` | DEPRECATED: (optional) Set true to create ServiceAccount for `prometheus-node-exporter` | `false` +`prometheus.prometheus-pushgateway.enabled` | DEPRECATED: (optional) Enable Prometheus `pushgateway` | `false` +`prometheus.prometheus-pushgateway.serviceAccount.create` | DEPRECATED: (optional) Set true to create ServiceAccount for `prometheus-pushgateway` | `false` +`prometheus.scrapeCAdvisor` | DEPRECATED: (optional) Enable Prometheus ScrapeCAdvisor | `false` +`prometheus.server.enabled` | (optional) If false, K10's Prometheus server will not be created, reducing the dashboard's functionality. | `true` +`prometheus.server.securityContext.runAsUser` | (optional) Set security context `runAsUser` ID for Prometheus server pod | `65534` +`prometheus.server.securityContext.runAsNonRoot` | (optional) Enable security context `runAsNonRoot` for Prometheus server pod | `true` +`prometheus.server.securityContext.runAsGroup` | (optional) Set security context `runAsGroup` ID for Prometheus server pod | `65534` +`prometheus.server.securityContext.fsGroup` | (optional) Set security context `fsGroup` ID for Prometheus server pod | `65534` +`prometheus.server.retention` | (optional) K10 Prometheus data retention | `"30d"` +`prometheus.server.strategy.rollingUpdate.maxSurge` | DEPRECATED: (optional) The number of Prometheus server pods that can be created above the desired amount of pods during an update | `"100%"` +`prometheus.server.strategy.rollingUpdate.maxUnavailable` | DEPRECATED: (optional) The number of Prometheus server pods that can be unavailable during the upgrade process | `"100%"` +`prometheus.server.strategy.type` | DEPRECATED: (optional) Change default deployment strategy for Prometheus server | `"RollingUpdate"` +`prometheus.server.persistentVolume.enabled` | DEPRECATED: (optional) If true, K10 Prometheus server will create a Persistent Volume Claim | `true` +`prometheus.server.persistentVolume.size` | (optional) K10 Prometheus server data Persistent Volume size | `30Gi` +`prometheus.server.persistentVolume.storageClass` | (optional) StorageClassName used to create Prometheus PVC. Setting this option overwrites global StorageClass value | `""` +`prometheus.server.configMapOverrideName` | DEPRECATED: (optional) Prometheus configmap name to override default generated name| `k10-prometheus-config` +`prometheus.server.fullnameOverride` | (optional) Prometheus deployment name to override default generated name| `prometheus-server` +`prometheus.server.baseURL` | (optional) K10 Prometheus external url path at which the server can be accessed | `/k10/prometheus/` +`prometheus.server.prefixURL` | (optional) K10 Prometheus prefix slug at which the server can be accessed | `/k10/prometheus/` +`prometheus.server.serviceAccounts.server.create` | DEPRECATED: (optional) Set true to create ServiceAccount for Prometheus server service | `true` +`grafana.enabled` | (optional) If false Grafana will not be available | `true` +`resources...[requests\|limits].[cpu\|memory]` | Overwriting the default K10 [container resource requests and limits](https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/) | varies depending on the container +`route.enabled` | Specifies whether the K10 dashboard should be exposed via route | `false` +`route.host` | FQDN (e.g., `.k10.example.com`) for name-based virtual host | `""` +`route.path` | URL path for K10 Dashboard (e.g., `/k10`) | `/` +`route.annotations` | Additional Route object annotations | `{}` +`route.labels` | Additional Route object labels | `{}` +`route.tls.enabled` | Configures a TLS use for `route.host` | `false` +`route.tls.insecureEdgeTerminationPolicy` | Specifies behavior for insecure scheme traffic | `Redirect` +`route.tls.termination` | Specifies the TLS termination of the route | `edge` +`apigateway.serviceResolver` | Specifies the resolver used for service discovery in the API gateway (`dns` or `endpoint`) | `dns` +`limiter.executorReplicas` | Specifies the number of executor-svc Pods used to process Kasten jobs | 3 +`limiter.executorThreads` | Specifies the number of threads per executor-svc Pod used to process Kasten jobs | 8 +`limiter.workloadSnapshotsPerAction` | Per action limit of concurrent manifest data snapshots, based on workload (ex. Namespace, Deployment, StatefulSet, VirtualMachine) | 5 +`limiter.csiSnapshotsPerCluster` | Cluster-wide limit of concurrent CSI VolumeSnapshot creation requests | `10` +`limiter.directSnapshotsPerCluster` | Cluster-wide limit of concurrent non-CSI snapshot creation requests | `10` +`limiter.snapshotExportsPerAction` | Per action limit of concurrent volume export operations | `3` +`limiter.snapshotExportsPerCluster` | Cluster-wide limit of concurrent volume export operations | `10` +`limiter.genericVolumeBackupsPerCluster` | Cluster-wide limit of concurrent Generic Volume Backup operations | `10` +`limiter.imageCopiesPerCluster` | Cluster-wide limit of concurrent ImageStream container image backup (i.e. copy from) and restore (i.e. copy to) operations | `10` +`limiter.workloadRestoresPerAction` | Per action limit of concurrent manifest data restores, based on workload (ex. Namespace, Deployment, StatefulSet, VirtualMachine) | 3 +`limiter.csiSnapshotRestoresPerAction` | Per action limit of concurrent CSI volume provisioning requests when restoring from VolumeSnapshots | 3 +`limiter.volumeRestoresPerAction` | Per action limit of concurrent volume restore operations from an exported backup | 3 +`limiter.volumeRestoresPerCluster` | Cluster-wide limit of concurrent volume restore operations from exported backups | `10` +`cluster.domainName` | Specifies the domain name of the cluster | `""` +`timeout.blueprintBackup` | Specifies the timeout (in minutes) for Blueprint backup actions | `45` +`timeout.blueprintRestore` | Specifies the timeout (in minutes) for Blueprint restore actions | `600` +`timeout.blueprintDelete` | Specifies the timeout (in minutes) for Blueprint delete actions | `45` +`timeout.blueprintHooks` | Specifies the timeout (in minutes) for Blueprint backupPrehook and backupPosthook actions | `20` +`timeout.checkRepoPodReady` | Specifies the timeout (in minutes) for temporary worker Pods used to validate backup repository existence | `20` +`timeout.statsPodReady` | Specifies the timeout (in minutes) for temporary worker Pods used to collect repository statistics | `20` +`timeout.efsRestorePodReady` | Specifies the timeout (in minutes) for temporary worker Pods used for shareable volume restore operations | `45` +`timeout.workerPodReady` | Specifies the timeout (in minutes) for all other temporary worker Pods used during Veeam Kasten operations | `15` +`timeout.jobWait` | Specifies the timeout (in minutes) for completing execution of any child job, after which the parent job will be canceled. If no value is set, a default of 10 hours will be used | `None` +`awsConfig.assumeRoleDuration` | Duration of a session token generated by AWS for an IAM role. The minimum value is 15 minutes and the maximum value is the maximum duration setting for that IAM role. For documentation about how to view and edit the maximum session duration for an IAM role see https://docs.aws.amazon.com/IAM/latest/UserGuide/id_roles_use.html#id_roles_use_view-role-max-session. The value accepts a number along with a single character ``m``(for minutes) or ``h`` (for hours) Examples: 60m or 2h | `''` +`awsConfig.efsBackupVaultName` | Specifies the AWS EFS backup vault name | `k10vault` +`vmWare.taskTimeoutMin` | Specifies the timeout for VMWare operations | `60` +`encryption.primaryKey.awsCmkKeyId` | Specifies the AWS CMK key ID for encrypting K10 Primary Key | `None` +`garbagecollector.daemonPeriod` | Sets garbage collection period (in seconds) | `21600` +`garbagecollector.keepMaxActions` | Sets maximum actions to keep | `1000` +`garbagecollector.actions.enabled` | Enables action collectors | `false` +`kubeVirtVMs.snapshot.unfreezeTimeout` | Defines the time duration within which the VMs must be unfrozen while backing them up. To know more about format [go doc](https://pkg.go.dev/time#ParseDuration) can be followed | `5m` +`excludedApps` | Specifies a list of applications to be excluded from the dashboard & compliance considerations. Format should be a :ref:`YAML array` | `["kube-system", "kube-ingress", "kube-node-lease", "kube-public", "kube-rook-ceph"]` +`workerPodMetricSidecar.enabled` | Enables a sidecar container for temporary worker Pods used to push Pod performance metrics to Prometheus | `true` +`workerPodMetricSidecar.metricLifetime` | Specifies the period after which metrics for an individual worker Pod are removed from Prometheus | `2m` +`workerPodMetricSidecar.pushGatewayInterval` | Specifies the frequency for pushing metrics into Prometheus | `30s` +`workerPodMetricSidecar.resources.[requests\|limits].[cpu\|memory]` | Specifies resource requests and limits for the temporary worker Pod metric sidecar | `{}` +`forceRootInBlueprintActions` | Forces any Pod created by a Blueprint to run as root user | `true` +`defaultPriorityClassName` | Specifies the default [priority class](https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/#priorityclass) name for all K10 deployments and ephemeral pods | `None` +`priorityClassName.` | Overrides the default [priority class](https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/#priorityclass) name for the specified deployment | `{}` +`ephemeralPVCOverhead` | Set the percentage increase for the ephemeral Persistent Volume Claim's storage request, e.g. PVC size = (file raw size) * (1 + `ephemeralPVCOverhead`) | `0.1` +`datastore.parallelUploads` | Specifies how many files can be uploaded in parallel to the data store | `8` +`datastore.parallelDownloads` | Specifies how many files can be downloaded in parallel from the data store | `8` +`kastenDisasterRecovery.quickMode.enabled` | Enables K10 Quick Disaster Recovery | `false` +`fips.enabled` | Specifies whether K10 should be run in the FIPS mode of operation | `false` +`workerPodCRDs.enabled` | Specifies whether K10 should use `ActionPodSpec` for granular resource control of worker pods | `false` +`workerPodCRDs.resourcesRequests.maxCPU` | Max CPU which might be setup in `ActionPodSpec` | `''` +`workerPodCRDs.resourcesRequests.maxMemory` | Max memory which might be setup in `ActionPodSpec` | `''` +`workerPodCRDs.defaultActionPodSpec.name` | The name of `ActionPodSpec` that will be used by default for worker pod resources. | `''` +`workerPodCRDs.defaultActionPodSpec.namespace` | The namespace of `ActionPodSpec` that will be used by default for worker pod resources. | `''` + + + +## Helm tips and tricks + +There is a way of setting values via a yaml file instead of using `--set`. +First, copy/paste values into a file (e.g., my_values.yaml): + +```yaml +secrets: + awsAccessKeyId: ${AWS_ACCESS_KEY_ID} + awsSecretAccessKey: ${AWS_SECRET_ACCESS_KEY} +``` + +and then run: + +```bash + envsubst < my_values.yaml > my_values_out.yaml && helm install k10 kasten/k10 -f my_values_out.yaml +``` + +To set a single value from a file, `--set-file` may be used over `--set`: + +```bash + helm install k10 kasten/k10 --set-file license=my_license.lic +``` + + +To use non-default GCP ServiceAccount (SA) credentials, the credentials JSON file needs to be encoded into a base64 +string: + +```bash + sa_key=$(base64 -w0 sa-key.json) + helm install k10 kasten/k10 --namespace=kasten-io --set secrets.googleApiKey=$sa_key +``` + +If the Google Service Account belongs to a project other than the one in which the cluster +is located, then the project's ID of the cluster must be also provided during the installation: + +```bash + sa_key=$(base64 -w0 sa-key.json) + helm install k10 kasten/k10 --namespace=kasten-io --set secrets.googleApiKey=$sa_key --set secrets.googleProjectId= +``` diff --git a/charts/kasten/k10/7.0.1401/app-readme.md b/charts/kasten/k10/7.0.1401/app-readme.md new file mode 100644 index 000000000..1b221891b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/app-readme.md @@ -0,0 +1,5 @@ +The K10 data management platform, purpose-built for Kubernetes, provides enterprise operations teams an easy-to-use, scalable, and secure system for backup/restore, disaster recovery, and mobility of Kubernetes applications. + +K10’s application-centric approach and deep integrations with relational and NoSQL databases, Kubernetes distributions, and all clouds provide teams the freedom of infrastructure choice without sacrificing operational simplicity. Policy-driven and extensible, K10 provides a native Kubernetes API and includes features such as full-spectrum consistency, database integrations, automatic application discovery, multi-cloud mobility, and a powerful web-based user interface. + +For more information, refer to the docs [https://docs.kasten.io/](https://docs.kasten.io/) diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/.helmignore b/charts/kasten/k10/7.0.1401/charts/grafana/.helmignore new file mode 100644 index 000000000..8cade1318 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/.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 +.vscode +.project +.idea/ +*.tmproj +OWNERS diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/Chart.yaml new file mode 100644 index 000000000..b5e9b92dc --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/Chart.yaml @@ -0,0 +1,35 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/grafana/helm-charts + - name: Upstream Project + url: https://github.com/grafana/grafana +apiVersion: v2 +appVersion: 11.2.2-security-01 +description: The leading tool for querying and visualizing time series and metrics. +home: https://grafana.com +icon: https://artifacthub.io/image/b4fed1a7-6c8f-4945-b99d-096efa3e4116 +keywords: +- monitoring +- metric +kubeVersion: ^1.8.0-0 +maintainers: +- email: zanhsieh@gmail.com + name: zanhsieh +- email: rluckie@cisco.com + name: rtluckie +- email: maor.friedman@redhat.com + name: maorfr +- email: miroslav.hadzhiev@gmail.com + name: Xtigyro +- email: mail@torstenwalter.de + name: torstenwalter +- email: github@jkroepke.de + name: jkroepke +name: grafana +sources: +- https://github.com/grafana/grafana +- https://github.com/grafana/helm-charts +type: application +version: 8.5.8 diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/README.md b/charts/kasten/k10/7.0.1401/charts/grafana/README.md new file mode 100644 index 000000000..4ab1a01c3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/README.md @@ -0,0 +1,783 @@ +# Grafana Helm Chart + +* Installs the web dashboarding system [Grafana](http://grafana.org/) + +## Get Repo Info + +```console +helm repo add grafana https://grafana.github.io/helm-charts +helm repo update +``` + +_See [helm repo](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Installing the Chart + +To install the chart with the release name `my-release`: + +```console +helm install my-release grafana/grafana +``` + +## Uninstalling the Chart + +To uninstall/delete the my-release deployment: + +```console +helm delete my-release +``` + +The command removes all the Kubernetes components associated with the chart and deletes the release. + +## Upgrading an existing Release to a new major version + +A major chart version change (like v1.2.3 -> v2.0.0) indicates that there is an +incompatible breaking change needing manual actions. + +### To 4.0.0 (And 3.12.1) + +This version requires Helm >= 2.12.0. + +### To 5.0.0 + +You have to add --force to your helm upgrade command as the labels of the chart have changed. + +### To 6.0.0 + +This version requires Helm >= 3.1.0. + +### To 7.0.0 + +For consistency with other Helm charts, the `global.image.registry` parameter was renamed +to `global.imageRegistry`. If you were not previously setting `global.image.registry`, no action +is required on upgrade. If you were previously setting `global.image.registry`, you will +need to instead set `global.imageRegistry`. + +## Configuration + +| Parameter | Description | Default | +|-------------------------------------------|-----------------------------------------------|---------------------------------------------------------| +| `replicas` | Number of nodes | `1` | +| `podDisruptionBudget.minAvailable` | Pod disruption minimum available | `nil` | +| `podDisruptionBudget.maxUnavailable` | Pod disruption maximum unavailable | `nil` | +| `podDisruptionBudget.apiVersion` | Pod disruption apiVersion | `nil` | +| `deploymentStrategy` | Deployment strategy | `{ "type": "RollingUpdate" }` | +| `livenessProbe` | Liveness Probe settings | `{ "httpGet": { "path": "/api/health", "port": 3000 } "initialDelaySeconds": 60, "timeoutSeconds": 30, "failureThreshold": 10 }` | +| `readinessProbe` | Readiness Probe settings | `{ "httpGet": { "path": "/api/health", "port": 3000 } }`| +| `securityContext` | Deployment securityContext | `{"runAsUser": 472, "runAsGroup": 472, "fsGroup": 472}` | +| `priorityClassName` | Name of Priority Class to assign pods | `nil` | +| `image.registry` | Image registry | `docker.io` | +| `image.repository` | Image repository | `grafana/grafana` | +| `image.tag` | Overrides the Grafana image tag whose default is the chart appVersion (`Must be >= 5.0.0`) | `` | +| `image.sha` | Image sha (optional) | `` | +| `image.pullPolicy` | Image pull policy | `IfNotPresent` | +| `image.pullSecrets` | Image pull secrets (can be templated) | `[]` | +| `service.enabled` | Enable grafana service | `true` | +| `service.ipFamilies` | Kubernetes service IP families | `[]` | +| `service.ipFamilyPolicy` | Kubernetes service IP family policy | `""` | +| `service.type` | Kubernetes service type | `ClusterIP` | +| `service.port` | Kubernetes port where service is exposed | `80` | +| `service.portName` | Name of the port on the service | `service` | +| `service.appProtocol` | Adds the appProtocol field to the service | `` | +| `service.targetPort` | Internal service is port | `3000` | +| `service.nodePort` | Kubernetes service nodePort | `nil` | +| `service.annotations` | Service annotations (can be templated) | `{}` | +| `service.labels` | Custom labels | `{}` | +| `service.clusterIP` | internal cluster service IP | `nil` | +| `service.loadBalancerIP` | IP address to assign to load balancer (if supported) | `nil` | +| `service.loadBalancerSourceRanges` | list of IP CIDRs allowed access to lb (if supported) | `[]` | +| `service.externalIPs` | service external IP addresses | `[]` | +| `service.externalTrafficPolicy` | change the default externalTrafficPolicy | `nil` | +| `headlessService` | Create a headless service | `false` | +| `extraExposePorts` | Additional service ports for sidecar containers| `[]` | +| `hostAliases` | adds rules to the pod's /etc/hosts | `[]` | +| `ingress.enabled` | Enables Ingress | `false` | +| `ingress.annotations` | Ingress annotations (values are templated) | `{}` | +| `ingress.labels` | Custom labels | `{}` | +| `ingress.path` | Ingress accepted path | `/` | +| `ingress.pathType` | Ingress type of path | `Prefix` | +| `ingress.hosts` | Ingress accepted hostnames | `["chart-example.local"]` | +| `ingress.extraPaths` | Ingress extra paths to prepend to every host configuration. Useful when configuring [custom actions with AWS ALB Ingress Controller](https://kubernetes-sigs.github.io/aws-load-balancer-controller/v2.6/guide/ingress/annotations/#actions). Requires `ingress.hosts` to have one or more host entries. | `[]` | +| `ingress.tls` | Ingress TLS configuration | `[]` | +| `ingress.ingressClassName` | Ingress Class Name. MAY be required for Kubernetes versions >= 1.18 | `""` | +| `resources` | CPU/Memory resource requests/limits | `{}` | +| `nodeSelector` | Node labels for pod assignment | `{}` | +| `tolerations` | Toleration labels for pod assignment | `[]` | +| `affinity` | Affinity settings for pod assignment | `{}` | +| `extraInitContainers` | Init containers to add to the grafana pod | `{}` | +| `extraContainers` | Sidecar containers to add to the grafana pod | `""` | +| `extraContainerVolumes` | Volumes that can be mounted in sidecar containers | `[]` | +| `extraLabels` | Custom labels for all manifests | `{}` | +| `schedulerName` | Name of the k8s scheduler (other than default) | `nil` | +| `persistence.enabled` | Use persistent volume to store data | `false` | +| `persistence.type` | Type of persistence (`pvc` or `statefulset`) | `pvc` | +| `persistence.size` | Size of persistent volume claim | `10Gi` | +| `persistence.existingClaim` | Use an existing PVC to persist data (can be templated) | `nil` | +| `persistence.storageClassName` | Type of persistent volume claim | `nil` | +| `persistence.accessModes` | Persistence access modes | `[ReadWriteOnce]` | +| `persistence.annotations` | PersistentVolumeClaim annotations | `{}` | +| `persistence.finalizers` | PersistentVolumeClaim finalizers | `[ "kubernetes.io/pvc-protection" ]` | +| `persistence.extraPvcLabels` | Extra labels to apply to a PVC. | `{}` | +| `persistence.subPath` | Mount a sub dir of the persistent volume (can be templated) | `nil` | +| `persistence.inMemory.enabled` | If persistence is not enabled, whether to mount the local storage in-memory to improve performance | `false` | +| `persistence.inMemory.sizeLimit` | SizeLimit for the in-memory local storage | `nil` | +| `persistence.disableWarning` | Hide NOTES warning, useful when persisting to a database | `false` | +| `initChownData.enabled` | If false, don't reset data ownership at startup | true | +| `initChownData.image.registry` | init-chown-data container image registry | `docker.io` | +| `initChownData.image.repository` | init-chown-data container image repository | `busybox` | +| `initChownData.image.tag` | init-chown-data container image tag | `1.31.1` | +| `initChownData.image.sha` | init-chown-data container image sha (optional)| `""` | +| `initChownData.image.pullPolicy` | init-chown-data container image pull policy | `IfNotPresent` | +| `initChownData.resources` | init-chown-data pod resource requests & limits | `{}` | +| `schedulerName` | Alternate scheduler name | `nil` | +| `env` | Extra environment variables passed to pods | `{}` | +| `envValueFrom` | Environment variables from alternate sources. See the API docs on [EnvVarSource](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.17/#envvarsource-v1-core) for format details. Can be templated | `{}` | +| `envFromSecret` | Name of a Kubernetes secret (must be manually created in the same namespace) containing values to be added to the environment. Can be templated | `""` | +| `envFromSecrets` | List of Kubernetes secrets (must be manually created in the same namespace) containing values to be added to the environment. Can be templated | `[]` | +| `envFromConfigMaps` | List of Kubernetes ConfigMaps (must be manually created in the same namespace) containing values to be added to the environment. Can be templated | `[]` | +| `envRenderSecret` | Sensible environment variables passed to pods and stored as secret. (passed through [tpl](https://helm.sh/docs/howto/charts_tips_and_tricks/#using-the-tpl-function)) | `{}` | +| `enableServiceLinks` | Inject Kubernetes services as environment variables. | `true` | +| `extraSecretMounts` | Additional grafana server secret mounts | `[]` | +| `extraVolumeMounts` | Additional grafana server volume mounts | `[]` | +| `extraVolumes` | Additional Grafana server volumes | `[]` | +| `automountServiceAccountToken` | Mounted the service account token on the grafana pod. Mandatory, if sidecars are enabled | `true` | +| `createConfigmap` | Enable creating the grafana configmap | `true` | +| `extraConfigmapMounts` | Additional grafana server configMap volume mounts (values are templated) | `[]` | +| `extraEmptyDirMounts` | Additional grafana server emptyDir volume mounts | `[]` | +| `plugins` | Plugins to be loaded along with Grafana | `[]` | +| `datasources` | Configure grafana datasources (passed through tpl) | `{}` | +| `alerting` | Configure grafana alerting (passed through tpl) | `{}` | +| `notifiers` | Configure grafana notifiers | `{}` | +| `dashboardProviders` | Configure grafana dashboard providers | `{}` | +| `dashboards` | Dashboards to import | `{}` | +| `dashboardsConfigMaps` | ConfigMaps reference that contains dashboards | `{}` | +| `grafana.ini` | Grafana's primary configuration | `{}` | +| `global.imageRegistry` | Global image pull registry for all images. | `null` | +| `global.imagePullSecrets` | Global image pull secrets (can be templated). Allows either an array of {name: pullSecret} maps (k8s-style), or an array of strings (more common helm-style). | `[]` | +| `ldap.enabled` | Enable LDAP authentication | `false` | +| `ldap.existingSecret` | The name of an existing secret containing the `ldap.toml` file, this must have the key `ldap-toml`. | `""` | +| `ldap.config` | Grafana's LDAP configuration | `""` | +| `annotations` | Deployment annotations | `{}` | +| `labels` | Deployment labels | `{}` | +| `podAnnotations` | Pod annotations | `{}` | +| `podLabels` | Pod labels | `{}` | +| `podPortName` | Name of the grafana port on the pod | `grafana` | +| `lifecycleHooks` | Lifecycle hooks for podStart and preStop [Example](https://kubernetes.io/docs/tasks/configure-pod-container/attach-handler-lifecycle-event/#define-poststart-and-prestop-handlers) | `{}` | +| `sidecar.image.registry` | Sidecar image registry | `quay.io` | +| `sidecar.image.repository` | Sidecar image repository | `kiwigrid/k8s-sidecar` | +| `sidecar.image.tag` | Sidecar image tag | `1.28.0` | +| `sidecar.image.sha` | Sidecar image sha (optional) | `""` | +| `sidecar.imagePullPolicy` | Sidecar image pull policy | `IfNotPresent` | +| `sidecar.resources` | Sidecar resources | `{}` | +| `sidecar.securityContext` | Sidecar securityContext | `{}` | +| `sidecar.enableUniqueFilenames` | Sets the kiwigrid/k8s-sidecar UNIQUE_FILENAMES environment variable. If set to `true` the sidecar will create unique filenames where duplicate data keys exist between ConfigMaps and/or Secrets within the same or multiple Namespaces. | `false` | +| `sidecar.alerts.enabled` | Enables the cluster wide search for alerts and adds/updates/deletes them in grafana |`false` | +| `sidecar.alerts.label` | Label that config maps with alerts should have to be added | `grafana_alert` | +| `sidecar.alerts.labelValue` | Label value that config maps with alerts should have to be added | `""` | +| `sidecar.alerts.searchNamespace` | Namespaces list. If specified, the sidecar will search for alerts config-maps inside these namespaces. Otherwise the namespace in which the sidecar is running will be used. It's also possible to specify ALL to search in all namespaces. | `nil` | +| `sidecar.alerts.watchMethod` | Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. | `WATCH` | +| `sidecar.alerts.resource` | Should the sidecar looks into secrets, configmaps or both. | `both` | +| `sidecar.alerts.reloadURL` | Full url of datasource configuration reload API endpoint, to invoke after a config-map change | `"http://localhost:3000/api/admin/provisioning/alerting/reload"` | +| `sidecar.alerts.skipReload` | Enabling this omits defining the REQ_URL and REQ_METHOD environment variables | `false` | +| `sidecar.alerts.initAlerts` | Set to true to deploy the alerts sidecar as an initContainer. This is needed if skipReload is true, to load any alerts defined at startup time. | `false` | +| `sidecar.alerts.extraMounts` | Additional alerts sidecar volume mounts. | `[]` | +| `sidecar.dashboards.enabled` | Enables the cluster wide search for dashboards and adds/updates/deletes them in grafana | `false` | +| `sidecar.dashboards.SCProvider` | Enables creation of sidecar provider | `true` | +| `sidecar.dashboards.provider.name` | Unique name of the grafana provider | `sidecarProvider` | +| `sidecar.dashboards.provider.orgid` | Id of the organisation, to which the dashboards should be added | `1` | +| `sidecar.dashboards.provider.folder` | Logical folder in which grafana groups dashboards | `""` | +| `sidecar.dashboards.provider.folderUid` | Allows you to specify the static UID for the logical folder above | `""` | +| `sidecar.dashboards.provider.disableDelete` | Activate to avoid the deletion of imported dashboards | `false` | +| `sidecar.dashboards.provider.allowUiUpdates` | Allow updating provisioned dashboards from the UI | `false` | +| `sidecar.dashboards.provider.type` | Provider type | `file` | +| `sidecar.dashboards.provider.foldersFromFilesStructure` | Allow Grafana to replicate dashboard structure from filesystem. | `false` | +| `sidecar.dashboards.watchMethod` | Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. | `WATCH` | +| `sidecar.skipTlsVerify` | Set to true to skip tls verification for kube api calls | `nil` | +| `sidecar.dashboards.label` | Label that config maps with dashboards should have to be added | `grafana_dashboard` | +| `sidecar.dashboards.labelValue` | Label value that config maps with dashboards should have to be added | `""` | +| `sidecar.dashboards.folder` | Folder in the pod that should hold the collected dashboards (unless `sidecar.dashboards.defaultFolderName` is set). This path will be mounted. | `/tmp/dashboards` | +| `sidecar.dashboards.folderAnnotation` | The annotation the sidecar will look for in configmaps to override the destination folder for files | `nil` | +| `sidecar.dashboards.defaultFolderName` | The default folder name, it will create a subfolder under the `sidecar.dashboards.folder` and put dashboards in there instead | `nil` | +| `sidecar.dashboards.searchNamespace` | Namespaces list. If specified, the sidecar will search for dashboards config-maps inside these namespaces. Otherwise the namespace in which the sidecar is running will be used. It's also possible to specify ALL to search in all namespaces. | `nil` | +| `sidecar.dashboards.script` | Absolute path to shell script to execute after a configmap got reloaded. | `nil` | +| `sidecar.dashboards.reloadURL` | Full url of dashboards configuration reload API endpoint, to invoke after a config-map change | `"http://localhost:3000/api/admin/provisioning/dashboards/reload"` | +| `sidecar.dashboards.skipReload` | Enabling this omits defining the REQ_USERNAME, REQ_PASSWORD, REQ_URL and REQ_METHOD environment variables | `false` | +| `sidecar.dashboards.resource` | Should the sidecar looks into secrets, configmaps or both. | `both` | +| `sidecar.dashboards.extraMounts` | Additional dashboard sidecar volume mounts. | `[]` | +| `sidecar.datasources.enabled` | Enables the cluster wide search for datasources and adds/updates/deletes them in grafana |`false` | +| `sidecar.datasources.label` | Label that config maps with datasources should have to be added | `grafana_datasource` | +| `sidecar.datasources.labelValue` | Label value that config maps with datasources should have to be added | `""` | +| `sidecar.datasources.searchNamespace` | Namespaces list. If specified, the sidecar will search for datasources config-maps inside these namespaces. Otherwise the namespace in which the sidecar is running will be used. It's also possible to specify ALL to search in all namespaces. | `nil` | +| `sidecar.datasources.watchMethod` | Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. | `WATCH` | +| `sidecar.datasources.resource` | Should the sidecar looks into secrets, configmaps or both. | `both` | +| `sidecar.datasources.reloadURL` | Full url of datasource configuration reload API endpoint, to invoke after a config-map change | `"http://localhost:3000/api/admin/provisioning/datasources/reload"` | +| `sidecar.datasources.skipReload` | Enabling this omits defining the REQ_URL and REQ_METHOD environment variables | `false` | +| `sidecar.datasources.initDatasources` | Set to true to deploy the datasource sidecar as an initContainer in addition to a container. This is needed if skipReload is true, to load any datasources defined at startup time. | `false` | +| `sidecar.notifiers.enabled` | Enables the cluster wide search for notifiers and adds/updates/deletes them in grafana | `false` | +| `sidecar.notifiers.label` | Label that config maps with notifiers should have to be added | `grafana_notifier` | +| `sidecar.notifiers.labelValue` | Label value that config maps with notifiers should have to be added | `""` | +| `sidecar.notifiers.searchNamespace` | Namespaces list. If specified, the sidecar will search for notifiers config-maps (or secrets) inside these namespaces. Otherwise the namespace in which the sidecar is running will be used. It's also possible to specify ALL to search in all namespaces. | `nil` | +| `sidecar.notifiers.watchMethod` | Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. | `WATCH` | +| `sidecar.notifiers.resource` | Should the sidecar looks into secrets, configmaps or both. | `both` | +| `sidecar.notifiers.reloadURL` | Full url of notifier configuration reload API endpoint, to invoke after a config-map change | `"http://localhost:3000/api/admin/provisioning/notifications/reload"` | +| `sidecar.notifiers.skipReload` | Enabling this omits defining the REQ_URL and REQ_METHOD environment variables | `false` | +| `sidecar.notifiers.initNotifiers` | Set to true to deploy the notifier sidecar as an initContainer in addition to a container. This is needed if skipReload is true, to load any notifiers defined at startup time. | `false` | +| `smtp.existingSecret` | The name of an existing secret containing the SMTP credentials. | `""` | +| `smtp.userKey` | The key in the existing SMTP secret containing the username. | `"user"` | +| `smtp.passwordKey` | The key in the existing SMTP secret containing the password. | `"password"` | +| `admin.existingSecret` | The name of an existing secret containing the admin credentials (can be templated). | `""` | +| `admin.userKey` | The key in the existing admin secret containing the username. | `"admin-user"` | +| `admin.passwordKey` | The key in the existing admin secret containing the password. | `"admin-password"` | +| `serviceAccount.automountServiceAccountToken` | Automount the service account token on all pods where is service account is used | `false` | +| `serviceAccount.annotations` | ServiceAccount annotations | | +| `serviceAccount.create` | Create service account | `true` | +| `serviceAccount.labels` | ServiceAccount labels | `{}` | +| `serviceAccount.name` | Service account name to use, when empty will be set to created account if `serviceAccount.create` is set else to `default` | `` | +| `serviceAccount.nameTest` | Service account name to use for test, when empty will be set to created account if `serviceAccount.create` is set else to `default` | `nil` | +| `rbac.create` | Create and use RBAC resources | `true` | +| `rbac.namespaced` | Creates Role and Rolebinding instead of the default ClusterRole and ClusteRoleBindings for the grafana instance | `false` | +| `rbac.useExistingRole` | Set to a rolename to use existing role - skipping role creating - but still doing serviceaccount and rolebinding to the rolename set here. | `nil` | +| `rbac.pspEnabled` | Create PodSecurityPolicy (with `rbac.create`, grant roles permissions as well) | `false` | +| `rbac.pspUseAppArmor` | Enforce AppArmor in created PodSecurityPolicy (requires `rbac.pspEnabled`) | `false` | +| `rbac.extraRoleRules` | Additional rules to add to the Role | [] | +| `rbac.extraClusterRoleRules` | Additional rules to add to the ClusterRole | [] | +| `command` | Define command to be executed by grafana container at startup | `nil` | +| `args` | Define additional args if command is used | `nil` | +| `testFramework.enabled` | Whether to create test-related resources | `true` | +| `testFramework.image.registry` | `test-framework` image registry. | `docker.io` | +| `testFramework.image.repository` | `test-framework` image repository. | `bats/bats` | +| `testFramework.image.tag` | `test-framework` image tag. | `v1.4.1` | +| `testFramework.imagePullPolicy` | `test-framework` image pull policy. | `IfNotPresent` | +| `testFramework.securityContext` | `test-framework` securityContext | `{}` | +| `downloadDashboards.env` | Environment variables to be passed to the `download-dashboards` container | `{}` | +| `downloadDashboards.envFromSecret` | Name of a Kubernetes secret (must be manually created in the same namespace) containing values to be added to the environment. Can be templated | `""` | +| `downloadDashboards.resources` | Resources of `download-dashboards` container | `{}` | +| `downloadDashboardsImage.registry` | Curl docker image registry | `docker.io` | +| `downloadDashboardsImage.repository` | Curl docker image repository | `curlimages/curl` | +| `downloadDashboardsImage.tag` | Curl docker image tag | `7.73.0` | +| `downloadDashboardsImage.sha` | Curl docker image sha (optional) | `""` | +| `downloadDashboardsImage.pullPolicy` | Curl docker image pull policy | `IfNotPresent` | +| `namespaceOverride` | Override the deployment namespace | `""` (`Release.Namespace`) | +| `serviceMonitor.enabled` | Use servicemonitor from prometheus operator | `false` | +| `serviceMonitor.namespace` | Namespace this servicemonitor is installed in | | +| `serviceMonitor.interval` | How frequently Prometheus should scrape | `1m` | +| `serviceMonitor.path` | Path to scrape | `/metrics` | +| `serviceMonitor.scheme` | Scheme to use for metrics scraping | `http` | +| `serviceMonitor.tlsConfig` | TLS configuration block for the endpoint | `{}` | +| `serviceMonitor.labels` | Labels for the servicemonitor passed to Prometheus Operator | `{}` | +| `serviceMonitor.scrapeTimeout` | Timeout after which the scrape is ended | `30s` | +| `serviceMonitor.relabelings` | RelabelConfigs to apply to samples before scraping. | `[]` | +| `serviceMonitor.metricRelabelings` | MetricRelabelConfigs to apply to samples before ingestion. | `[]` | +| `revisionHistoryLimit` | Number of old ReplicaSets to retain | `10` | +| `imageRenderer.enabled` | Enable the image-renderer deployment & service | `false` | +| `imageRenderer.image.registry` | image-renderer Image registry | `docker.io` | +| `imageRenderer.image.repository` | image-renderer Image repository | `grafana/grafana-image-renderer` | +| `imageRenderer.image.tag` | image-renderer Image tag | `latest` | +| `imageRenderer.image.sha` | image-renderer Image sha (optional) | `""` | +| `imageRenderer.image.pullPolicy` | image-renderer ImagePullPolicy | `Always` | +| `imageRenderer.env` | extra env-vars for image-renderer | `{}` | +| `imageRenderer.envValueFrom` | Environment variables for image-renderer from alternate sources. See the API docs on [EnvVarSource](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.17/#envvarsource-v1-core) for format details. Can be templated | `{}` | +| `imageRenderer.extraConfigmapMounts` | Additional image-renderer configMap volume mounts (values are templated) | `[]` | +| `imageRenderer.extraSecretMounts` | Additional image-renderer secret volume mounts | `[]` | +| `imageRenderer.extraVolumeMounts` | Additional image-renderer volume mounts | `[]` | +| `imageRenderer.extraVolumes` | Additional image-renderer volumes | `[]` | +| `imageRenderer.serviceAccountName` | image-renderer deployment serviceAccountName | `""` | +| `imageRenderer.securityContext` | image-renderer deployment securityContext | `{}` | +| `imageRenderer.podAnnotations` | image-renderer image-renderer pod annotation | `{}` | +| `imageRenderer.hostAliases` | image-renderer deployment Host Aliases | `[]` | +| `imageRenderer.priorityClassName` | image-renderer deployment priority class | `''` | +| `imageRenderer.service.enabled` | Enable the image-renderer service | `true` | +| `imageRenderer.service.portName` | image-renderer service port name | `http` | +| `imageRenderer.service.port` | image-renderer port used by deployment | `8081` | +| `imageRenderer.service.targetPort` | image-renderer service port used by service | `8081` | +| `imageRenderer.appProtocol` | Adds the appProtocol field to the service | `` | +| `imageRenderer.grafanaSubPath` | Grafana sub path to use for image renderer callback url | `''` | +| `imageRenderer.serverURL` | Remote image renderer url | `''` | +| `imageRenderer.renderingCallbackURL` | Callback url for the Grafana image renderer | `''` | +| `imageRenderer.podPortName` | name of the image-renderer port on the pod | `http` | +| `imageRenderer.revisionHistoryLimit` | number of image-renderer replica sets to keep | `10` | +| `imageRenderer.networkPolicy.limitIngress` | Enable a NetworkPolicy to limit inbound traffic from only the created grafana pods | `true` | +| `imageRenderer.networkPolicy.limitEgress` | Enable a NetworkPolicy to limit outbound traffic to only the created grafana pods | `false` | +| `imageRenderer.resources` | Set resource limits for image-renderer pods | `{}` | +| `imageRenderer.nodeSelector` | Node labels for pod assignment | `{}` | +| `imageRenderer.tolerations` | Toleration labels for pod assignment | `[]` | +| `imageRenderer.affinity` | Affinity settings for pod assignment | `{}` | +| `networkPolicy.enabled` | Enable creation of NetworkPolicy resources. | `false` | +| `networkPolicy.allowExternal` | Don't require client label for connections | `true` | +| `networkPolicy.explicitNamespacesSelector` | A Kubernetes LabelSelector to explicitly select namespaces from which traffic could be allowed | `{}` | +| `networkPolicy.ingress` | Enable the creation of an ingress network policy | `true` | +| `networkPolicy.egress.enabled` | Enable the creation of an egress network policy | `false` | +| `networkPolicy.egress.ports` | An array of ports to allow for the egress | `[]` | +| `enableKubeBackwardCompatibility` | Enable backward compatibility of kubernetes where pod's defintion version below 1.13 doesn't have the enableServiceLinks option | `false` | + +### Example ingress with path + +With grafana 6.3 and above + +```yaml +grafana.ini: + server: + domain: monitoring.example.com + root_url: "%(protocol)s://%(domain)s/grafana" + serve_from_sub_path: true +ingress: + enabled: true + hosts: + - "monitoring.example.com" + path: "/grafana" +``` + +### Example of extraVolumeMounts and extraVolumes + +Configure additional volumes with `extraVolumes` and volume mounts with `extraVolumeMounts`. + +Example for `extraVolumeMounts` and corresponding `extraVolumes`: + +```yaml +extraVolumeMounts: + - name: plugins + mountPath: /var/lib/grafana/plugins + subPath: configs/grafana/plugins + readOnly: false + - name: dashboards + mountPath: /var/lib/grafana/dashboards + hostPath: /usr/shared/grafana/dashboards + readOnly: false + +extraVolumes: + - name: plugins + existingClaim: existing-grafana-claim + - name: dashboards + hostPath: /usr/shared/grafana/dashboards +``` + +Volumes default to `emptyDir`. Set to `persistentVolumeClaim`, +`hostPath`, `csi`, or `configMap` for other types. For a +`persistentVolumeClaim`, specify an existing claim name with +`existingClaim`. + +## Import dashboards + +There are a few methods to import dashboards to Grafana. Below are some examples and explanations as to how to use each method: + +```yaml +dashboards: + default: + some-dashboard: + json: | + { + "annotations": + + ... + # Complete json file here + ... + + "title": "Some Dashboard", + "uid": "abcd1234", + "version": 1 + } + custom-dashboard: + # This is a path to a file inside the dashboards directory inside the chart directory + file: dashboards/custom-dashboard.json + prometheus-stats: + # Ref: https://grafana.com/dashboards/2 + gnetId: 2 + revision: 2 + datasource: Prometheus + loki-dashboard-quick-search: + gnetId: 12019 + revision: 2 + datasource: + - name: DS_PROMETHEUS + value: Prometheus + - name: DS_LOKI + value: Loki + local-dashboard: + url: https://raw.githubusercontent.com/user/repository/master/dashboards/dashboard.json +``` + +## BASE64 dashboards + +Dashboards could be stored on a server that does not return JSON directly and instead of it returns a Base64 encoded file (e.g. Gerrit) +A new parameter has been added to the url use case so if you specify a b64content value equals to true after the url entry a Base64 decoding is applied before save the file to disk. +If this entry is not set or is equals to false not decoding is applied to the file before saving it to disk. + +### Gerrit use case + +Gerrit API for download files has the following schema: where {project-name} and +{file-id} usually has '/' in their values and so they MUST be replaced by %2F so if project-name is user/repo, branch-id is master and file-id is equals to dir1/dir2/dashboard +the url value is + +## Sidecar for dashboards + +If the parameter `sidecar.dashboards.enabled` is set, a sidecar container is deployed in the grafana +pod. This container watches all configmaps (or secrets) in the cluster and filters out the ones with +a label as defined in `sidecar.dashboards.label`. The files defined in those configmaps are written +to a folder and accessed by grafana. Changes to the configmaps are monitored and the imported +dashboards are deleted/updated. + +A recommendation is to use one configmap per dashboard, as a reduction of multiple dashboards inside +one configmap is currently not properly mirrored in grafana. + +Example dashboard config: + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: sample-grafana-dashboard + labels: + grafana_dashboard: "1" +data: + k8s-dashboard.json: |- + [...] +``` + +## Sidecar for datasources + +If the parameter `sidecar.datasources.enabled` is set, an init container is deployed in the grafana +pod. This container lists all secrets (or configmaps, though not recommended) in the cluster and +filters out the ones with a label as defined in `sidecar.datasources.label`. The files defined in +those secrets are written to a folder and accessed by grafana on startup. Using these yaml files, +the data sources in grafana can be imported. + +Should you aim for reloading datasources in Grafana each time the config is changed, set `sidecar.datasources.skipReload: false` and adjust `sidecar.datasources.reloadURL` to `http://..svc.cluster.local/api/admin/provisioning/datasources/reload`. + +Secrets are recommended over configmaps for this usecase because datasources usually contain private +data like usernames and passwords. Secrets are the more appropriate cluster resource to manage those. + +Example values to add a postgres datasource as a kubernetes secret: + +```yaml +apiVersion: v1 +kind: Secret +metadata: + name: grafana-datasources + labels: + grafana_datasource: 'true' # default value for: sidecar.datasources.label +stringData: + pg-db.yaml: |- + apiVersion: 1 + datasources: + - name: My pg db datasource + type: postgres + url: my-postgresql-db:5432 + user: db-readonly-user + secureJsonData: + password: 'SUperSEcretPa$$word' + jsonData: + database: my_datase + sslmode: 'disable' # disable/require/verify-ca/verify-full + maxOpenConns: 0 # Grafana v5.4+ + maxIdleConns: 2 # Grafana v5.4+ + connMaxLifetime: 14400 # Grafana v5.4+ + postgresVersion: 1000 # 903=9.3, 904=9.4, 905=9.5, 906=9.6, 1000=10 + timescaledb: false + # allow users to edit datasources from the UI. + editable: false +``` + +Example values to add a datasource adapted from [Grafana](http://docs.grafana.org/administration/provisioning/#example-datasource-config-file): + +```yaml +datasources: + datasources.yaml: + apiVersion: 1 + datasources: + # name of the datasource. Required + - name: Graphite + # datasource type. Required + type: graphite + # access mode. proxy or direct (Server or Browser in the UI). Required + access: proxy + # org id. will default to orgId 1 if not specified + orgId: 1 + # url + url: http://localhost:8080 + # database password, if used + password: + # database user, if used + user: + # database name, if used + database: + # enable/disable basic auth + basicAuth: + # basic auth username + basicAuthUser: + # basic auth password + basicAuthPassword: + # enable/disable with credentials headers + withCredentials: + # mark as default datasource. Max one per org + isDefault: + # fields that will be converted to json and stored in json_data + jsonData: + graphiteVersion: "1.1" + tlsAuth: true + tlsAuthWithCACert: true + # json object of data that will be encrypted. + secureJsonData: + tlsCACert: "..." + tlsClientCert: "..." + tlsClientKey: "..." + version: 1 + # allow users to edit datasources from the UI. + editable: false +``` + +## Sidecar for notifiers + +If the parameter `sidecar.notifiers.enabled` is set, an init container is deployed in the grafana +pod. This container lists all secrets (or configmaps, though not recommended) in the cluster and +filters out the ones with a label as defined in `sidecar.notifiers.label`. The files defined in +those secrets are written to a folder and accessed by grafana on startup. Using these yaml files, +the notification channels in grafana can be imported. The secrets must be created before +`helm install` so that the notifiers init container can list the secrets. + +Secrets are recommended over configmaps for this usecase because alert notification channels usually contain +private data like SMTP usernames and passwords. Secrets are the more appropriate cluster resource to manage those. + +Example datasource config adapted from [Grafana](https://grafana.com/docs/grafana/latest/administration/provisioning/#alert-notification-channels): + +```yaml +notifiers: + - name: notification-channel-1 + type: slack + uid: notifier1 + # either + org_id: 2 + # or + org_name: Main Org. + is_default: true + send_reminder: true + frequency: 1h + disable_resolve_message: false + # See `Supported Settings` section for settings supporter for each + # alert notification type. + settings: + recipient: 'XXX' + token: 'xoxb' + uploadImage: true + url: https://slack.com + +delete_notifiers: + - name: notification-channel-1 + uid: notifier1 + org_id: 2 + - name: notification-channel-2 + # default org_id: 1 +``` + +## Sidecar for alerting resources + +If the parameter `sidecar.alerts.enabled` is set, a sidecar container is deployed in the grafana +pod. This container watches all configmaps (or secrets) in the cluster (namespace defined by `sidecar.alerts.searchNamespace`) and filters out the ones with +a label as defined in `sidecar.alerts.label` (default is `grafana_alert`). The files defined in those configmaps are written +to a folder and accessed by grafana. Changes to the configmaps are monitored and the imported alerting resources are updated, however, deletions are a little more complicated (see below). + +This sidecar can be used to provision alert rules, contact points, notification policies, notification templates and mute timings as shown in [Grafana Documentation](https://grafana.com/docs/grafana/next/alerting/set-up/provision-alerting-resources/file-provisioning/). + +To fetch the alert config which will be provisioned, use the alert provisioning API ([Grafana Documentation](https://grafana.com/docs/grafana/next/developers/http_api/alerting_provisioning/)). +You can use either JSON or YAML format. + +Example config for an alert rule: + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: sample-grafana-alert + labels: + grafana_alert: "1" +data: + k8s-alert.yml: |- + apiVersion: 1 + groups: + - orgId: 1 + name: k8s-alert + [...] +``` + +To delete provisioned alert rules is a two step process, you need to delete the configmap which defined the alert rule +and then create a configuration which deletes the alert rule. + +Example deletion configuration: + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: delete-sample-grafana-alert + namespace: monitoring + labels: + grafana_alert: "1" +data: + delete-k8s-alert.yml: |- + apiVersion: 1 + deleteRules: + - orgId: 1 + uid: 16624780-6564-45dc-825c-8bded4ad92d3 +``` + +## Statically provision alerting resources + +If you don't need to change alerting resources (alert rules, contact points, notification policies and notification templates) regularly you could use the `alerting` config option instead of the sidecar option above. +This will grab the alerting config and apply it statically at build time for the helm file. + +There are two methods to statically provision alerting configuration in Grafana. Below are some examples and explanations as to how to use each method: + +```yaml +alerting: + team1-alert-rules.yaml: + file: alerting/team1/rules.yaml + team2-alert-rules.yaml: + file: alerting/team2/rules.yaml + team3-alert-rules.yaml: + file: alerting/team3/rules.yaml + notification-policies.yaml: + file: alerting/shared/notification-policies.yaml + notification-templates.yaml: + file: alerting/shared/notification-templates.yaml + contactpoints.yaml: + apiVersion: 1 + contactPoints: + - orgId: 1 + name: Slack channel + receivers: + - uid: default-receiver + type: slack + settings: + # Webhook URL to be filled in + url: "" + # We need to escape double curly braces for the tpl function. + text: '{{ `{{ template "default.message" . }}` }}' + title: '{{ `{{ template "default.title" . }}` }}' +``` + +The two possibilities for static alerting resource provisioning are: + +* Inlining the file contents as shown for contact points in the above example. +* Importing a file using a relative path starting from the chart root directory as shown for the alert rules in the above example. + +### Important notes on file provisioning + +* The format of the files is defined in the [Grafana documentation](https://grafana.com/docs/grafana/next/alerting/set-up/provision-alerting-resources/file-provisioning/) on file provisioning. +* The chart supports importing YAML and JSON files. +* The filename must be unique, otherwise one volume mount will overwrite the other. +* In case of inlining, double curly braces that arise from the Grafana configuration format and are not intended as templates for the chart must be escaped. +* The number of total files under `alerting:` is not limited. Each file will end up as a volume mount in the corresponding provisioning folder of the deployed Grafana instance. +* The file size for each import is limited by what the function `.Files.Get` can handle, which suffices for most cases. + +## How to serve Grafana with a path prefix (/grafana) + +In order to serve Grafana with a prefix (e.g., ), add the following to your values.yaml. + +```yaml +ingress: + enabled: true + annotations: + kubernetes.io/ingress.class: "nginx" + nginx.ingress.kubernetes.io/rewrite-target: /$1 + nginx.ingress.kubernetes.io/use-regex: "true" + + path: /grafana/?(.*) + hosts: + - k8s.example.dev + +grafana.ini: + server: + root_url: http://localhost:3000/grafana # this host can be localhost +``` + +## How to securely reference secrets in grafana.ini + +This example uses Grafana [file providers](https://grafana.com/docs/grafana/latest/administration/configuration/#file-provider) for secret values and the `extraSecretMounts` configuration flag (Additional grafana server secret mounts) to mount the secrets. + +In grafana.ini: + +```yaml +grafana.ini: + [auth.generic_oauth] + enabled = true + client_id = $__file{/etc/secrets/auth_generic_oauth/client_id} + client_secret = $__file{/etc/secrets/auth_generic_oauth/client_secret} +``` + +Existing secret, or created along with helm: + +```yaml +--- +apiVersion: v1 +kind: Secret +metadata: + name: auth-generic-oauth-secret +type: Opaque +stringData: + client_id: + client_secret: +``` + +Include in the `extraSecretMounts` configuration flag: + +```yaml +extraSecretMounts: + - name: auth-generic-oauth-secret-mount + secretName: auth-generic-oauth-secret + defaultMode: 0440 + mountPath: /etc/secrets/auth_generic_oauth + readOnly: true +``` + +### extraSecretMounts using a Container Storage Interface (CSI) provider + +This example uses a CSI driver e.g. retrieving secrets using [Azure Key Vault Provider](https://github.com/Azure/secrets-store-csi-driver-provider-azure) + +```yaml +extraSecretMounts: + - name: secrets-store-inline + mountPath: /run/secrets + readOnly: true + csi: + driver: secrets-store.csi.k8s.io + readOnly: true + volumeAttributes: + secretProviderClass: "my-provider" + nodePublishSecretRef: + name: akv-creds +``` + +## Image Renderer Plug-In + +This chart supports enabling [remote image rendering](https://github.com/grafana/grafana-image-renderer/blob/master/README.md#run-in-docker) + +```yaml +imageRenderer: + enabled: true +``` + +### Image Renderer NetworkPolicy + +By default the image-renderer pods will have a network policy which only allows ingress traffic from the created grafana instance + +### High Availability for unified alerting + +If you want to run Grafana in a high availability cluster you need to enable +the headless service by setting `headlessService: true` in your `values.yaml` +file. + +As next step you have to setup the `grafana.ini` in your `values.yaml` in a way +that it will make use of the headless service to obtain all the IPs of the +cluster. You should replace ``{{ Name }}`` with the name of your helm deployment. + +```yaml +grafana.ini: + ... + unified_alerting: + enabled: true + ha_peers: {{ Name }}-headless:9094 + ha_listen_address: ${POD_IP}:9094 + ha_advertise_address: ${POD_IP}:9094 + + alerting: + enabled: false +``` diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/default-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/default-values.yaml new file mode 100644 index 000000000..fc2ba605a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/default-values.yaml @@ -0,0 +1 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-affinity-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-affinity-values.yaml new file mode 100644 index 000000000..f5b9b53e7 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-affinity-values.yaml @@ -0,0 +1,16 @@ +affinity: + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchLabels: + app.kubernetes.io/instance: grafana-test + app.kubernetes.io/name: grafana + topologyKey: failure-domain.beta.kubernetes.io/zone + weight: 100 + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: + app.kubernetes.io/instance: grafana-test + app.kubernetes.io/name: grafana + topologyKey: kubernetes.io/hostname diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-json-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-json-values.yaml new file mode 100644 index 000000000..e0c4e4168 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-json-values.yaml @@ -0,0 +1,53 @@ +dashboards: + my-provider: + my-awesome-dashboard: + # An empty but valid dashboard + json: | + { + "__inputs": [], + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "6.3.5" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": "-- Grafana --", + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "type": "dashboard" + } + ] + }, + "editable": true, + "gnetId": null, + "graphTooltip": 0, + "id": null, + "links": [], + "panels": [], + "schemaVersion": 19, + "style": "dark", + "tags": [], + "templating": { + "list": [] + }, + "time": { + "from": "now-6h", + "to": "now" + }, + "timepicker": { + "refresh_intervals": ["5s"] + }, + "timezone": "", + "title": "Dummy Dashboard", + "uid": "IdcYQooWk", + "version": 1 + } + datasource: Prometheus diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-values.yaml new file mode 100644 index 000000000..7b662c5fd --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-dashboard-values.yaml @@ -0,0 +1,19 @@ +dashboards: + my-provider: + my-awesome-dashboard: + gnetId: 10000 + revision: 1 + datasource: Prometheus +dashboardProviders: + dashboardproviders.yaml: + apiVersion: 1 + providers: + - name: 'my-provider' + orgId: 1 + folder: '' + type: file + updateIntervalSeconds: 10 + disableDeletion: true + editable: true + options: + path: /var/lib/grafana/dashboards/my-provider diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-extraconfigmapmounts-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-extraconfigmapmounts-values.yaml new file mode 100644 index 000000000..5cc44a056 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-extraconfigmapmounts-values.yaml @@ -0,0 +1,7 @@ +extraConfigmapMounts: + - name: '{{ include "grafana.fullname" . }}' + configMap: '{{ include "grafana.fullname" . }}' + mountPath: /var/lib/grafana/dashboards/test-dashboard.json + # This is not a realistic test, but for this we only care about extraConfigmapMounts not being empty and pointing to an existing ConfigMap + subPath: grafana.ini + readOnly: true diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-image-renderer-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-image-renderer-values.yaml new file mode 100644 index 000000000..06c0bda13 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-image-renderer-values.yaml @@ -0,0 +1,107 @@ +podLabels: + customLableA: Aaaaa +imageRenderer: + enabled: true + env: + RENDERING_ARGS: --disable-gpu,--window-size=1280x758 + RENDERING_MODE: clustered + podLabels: + customLableB: Bbbbb + networkPolicy: + limitIngress: true + limitEgress: true + resources: + limits: + cpu: 1000m + memory: 1000Mi + requests: + cpu: 500m + memory: 50Mi + extraVolumes: + - name: empty-renderer-volume + emtpyDir: {} + extraVolumeMounts: + - mountPath: /tmp/renderer + name: empty-renderer-volume + extraConfigmapMounts: + - name: renderer-config + mountPath: /usr/src/app/config.json + subPath: renderer-config.json + configMap: image-renderer-config + extraSecretMounts: + - name: renderer-certificate + mountPath: /usr/src/app/certs/ + secretName: image-renderer-certificate + readOnly: true + +extraObjects: + - apiVersion: v1 + kind: ConfigMap + metadata: + name: image-renderer-config + data: + renderer-config.json: | + { + "service": { + "host": null, + "port": 8081, + "protocol": "http", + "certFile": "", + "certKey": "", + + "metrics": { + "enabled": true, + "collectDefaultMetrics": true, + "requestDurationBuckets": [1, 5, 7, 9, 11, 13, 15, 20, 30] + }, + + "logging": { + "level": "info", + "console": { + "json": true, + "colorize": false + } + }, + + "security": { + "authToken": "-" + } + }, + "rendering": { + "chromeBin": null, + "args": ["--no-sandbox", "--disable-gpu"], + "ignoresHttpsErrors": false, + + "timezone": null, + "acceptLanguage": null, + "width": 1000, + "height": 500, + "deviceScaleFactor": 1, + "maxWidth": 3080, + "maxHeight": 3000, + "maxDeviceScaleFactor": 4, + "pageZoomLevel": 1, + "headed": false, + + "mode": "default", + "emulateNetworkConditions": false, + "clustering": { + "monitor": false, + "mode": "browser", + "maxConcurrency": 5, + "timeout": 30 + }, + + "verboseLogging": false, + "dumpio": false, + "timingMetrics": false + } + } + - apiVersion: v1 + kind: Secret + metadata: + name: image-renderer-certificate + type: Opaque + data: + # Decodes to 'PLACEHOLDER CERTIFICATE' + not-a-real-certificate: UExBQ0VIT0xERVIgQ0VSVElGSUNBVEU= diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-nondefault-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-nondefault-values.yaml new file mode 100644 index 000000000..fb5c17940 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-nondefault-values.yaml @@ -0,0 +1,6 @@ +global: + environment: prod +ingress: + enabled: true + hosts: + - monitoring-{{ .Values.global.environment }}.example.com diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-persistence.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-persistence.yaml new file mode 100644 index 000000000..b92ca02c9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-persistence.yaml @@ -0,0 +1,3 @@ +persistence: + type: pvc + enabled: true diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml new file mode 100644 index 000000000..a6935e56d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml @@ -0,0 +1,38 @@ +extraObjects: + - apiVersion: v1 + kind: ConfigMap + metadata: + name: '{{ include "grafana.fullname" . }}-test' + data: + var1: "value1" + - apiVersion: v1 + kind: Secret + metadata: + name: '{{ include "grafana.fullname" . }}-test' + type: Opaque + data: + var2: "dmFsdWUy" + +sidecar: + dashboards: + enabled: true + envValueFrom: + VAR1: + configMapKeyRef: + name: '{{ include "grafana.fullname" . }}-test' + key: var1 + VAR2: + secretKeyRef: + name: '{{ include "grafana.fullname" . }}-test' + key: var2 + datasources: + enabled: true + envValueFrom: + VAR1: + configMapKeyRef: + name: '{{ include "grafana.fullname" . }}-test' + key: var1 + VAR2: + secretKeyRef: + name: '{{ include "grafana.fullname" . }}-test' + key: var2 diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/dashboards/custom-dashboard.json b/charts/kasten/k10/7.0.1401/charts/grafana/dashboards/custom-dashboard.json new file mode 100644 index 000000000..9e26dfeeb --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/dashboards/custom-dashboard.json @@ -0,0 +1 @@ +{} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/grafana/templates/NOTES.txt new file mode 100644 index 000000000..a40f666a4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/NOTES.txt @@ -0,0 +1,55 @@ +1. Get your '{{ .Values.adminUser }}' user password by running: + + kubectl get secret --namespace {{ include "grafana.namespace" . }} {{ .Values.admin.existingSecret | default (include "grafana.fullname" .) }} -o jsonpath="{.data.{{ .Values.admin.passwordKey | default "admin-password" }}}" | base64 --decode ; echo + + +2. The Grafana server can be accessed via port {{ .Values.service.port }} on the following DNS name from within your cluster: + + {{ include "grafana.fullname" . }}.{{ include "grafana.namespace" . }}.svc.cluster.local +{{ if .Values.ingress.enabled }} + If you bind grafana to 80, please update values in values.yaml and reinstall: + ``` + securityContext: + runAsUser: 0 + runAsGroup: 0 + fsGroup: 0 + + command: + - "setcap" + - "'cap_net_bind_service=+ep'" + - "/usr/sbin/grafana-server &&" + - "sh" + - "/run.sh" + ``` + Details refer to https://grafana.com/docs/installation/configuration/#http-port. + Or grafana would always crash. + + From outside the cluster, the server URL(s) are: + {{- range .Values.ingress.hosts }} + http://{{ . }} + {{- end }} +{{- else }} + Get the Grafana URL to visit by running these commands in the same shell: + {{- if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ include "grafana.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "grafana.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ include "grafana.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT + {{- else if contains "LoadBalancer" .Values.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ include "grafana.namespace" . }} -w {{ include "grafana.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ include "grafana.namespace" . }} {{ include "grafana.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + http://$SERVICE_IP:{{ .Values.service.port -}} + {{- else if contains "ClusterIP" .Values.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ include "grafana.namespace" . }} -l "app.kubernetes.io/name={{ include "grafana.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ include "grafana.namespace" . }} port-forward $POD_NAME 3000 + {{- end }} +{{- end }} + +3. Login with the password from step 1 and the username: {{ .Values.adminUser }} + +{{- if and (not .Values.persistence.enabled) (not .Values.persistence.disableWarning) }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the Grafana pod is terminated. ##### +################################################################################# +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/_config.tpl b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_config.tpl new file mode 100644 index 000000000..889762006 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_config.tpl @@ -0,0 +1,176 @@ +{{/* + Generate config map data + */}} +{{- define "grafana.configData" -}} +{{ include "grafana.assertNoLeakedSecrets" . }} +{{- $files := .Files }} +{{- $root := . -}} +{{- with .Values.plugins }} +plugins: {{ join "," . }} +{{- end }} +grafana.ini: | +{{- range $elem, $elemVal := index .Values "grafana.ini" }} + {{- if not (kindIs "map" $elemVal) }} + {{- if kindIs "invalid" $elemVal }} + {{ $elem }} = + {{- else if kindIs "slice" $elemVal }} + {{ $elem }} = {{ toJson $elemVal }} + {{- else if kindIs "string" $elemVal }} + {{ $elem }} = {{ tpl $elemVal $ }} + {{- else }} + {{ $elem }} = {{ $elemVal }} + {{- end }} + {{- end }} +{{- end }} +{{- range $key, $value := index .Values "grafana.ini" }} + {{- if kindIs "map" $value }} + [{{ $key }}] + {{- range $elem, $elemVal := $value }} + {{- if kindIs "invalid" $elemVal }} + {{ $elem }} = + {{- else if kindIs "slice" $elemVal }} + {{ $elem }} = {{ toJson $elemVal }} + {{- else if kindIs "string" $elemVal }} + {{ $elem }} = {{ tpl $elemVal $ }} + {{- else }} + {{ $elem }} = {{ $elemVal }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} + +{{- range $key, $value := .Values.datasources }} +{{- if not (hasKey $value "secret") }} +{{ $key }}: | + {{- tpl (toYaml $value | nindent 2) $root }} +{{- end }} +{{- end }} + +{{- range $key, $value := .Values.notifiers }} +{{- if not (hasKey $value "secret") }} +{{ $key }}: | + {{- toYaml $value | nindent 2 }} +{{- end }} +{{- end }} + +{{- range $key, $value := .Values.alerting }} +{{- if (hasKey $value "file") }} +{{ $key }}: +{{- toYaml ( $files.Get $value.file ) | nindent 2 }} +{{- else if (or (hasKey $value "secret") (hasKey $value "secretFile"))}} +{{/* will be stored inside secret generated by "configSecret.yaml"*/}} +{{- else }} +{{ $key }}: | + {{- tpl (toYaml $value | nindent 2) $root }} +{{- end }} +{{- end }} + +{{- range $key, $value := .Values.dashboardProviders }} +{{ $key }}: | + {{- toYaml $value | nindent 2 }} +{{- end }} + +{{- if .Values.dashboards }} +download_dashboards.sh: | + #!/usr/bin/env sh + set -euf + {{- if .Values.dashboardProviders }} + {{- range $key, $value := .Values.dashboardProviders }} + {{- range $value.providers }} + mkdir -p {{ .options.path }} + {{- end }} + {{- end }} + {{- end }} +{{ $dashboardProviders := .Values.dashboardProviders }} +{{- range $provider, $dashboards := .Values.dashboards }} + {{- range $key, $value := $dashboards }} + {{- if (or (hasKey $value "gnetId") (hasKey $value "url")) }} + curl -skf \ + --connect-timeout 60 \ + --max-time 60 \ + {{- if not $value.b64content }} + {{- if not $value.acceptHeader }} + -H "Accept: application/json" \ + {{- else }} + -H "Accept: {{ $value.acceptHeader }}" \ + {{- end }} + {{- if $value.token }} + -H "Authorization: token {{ $value.token }}" \ + {{- end }} + {{- if $value.bearerToken }} + -H "Authorization: Bearer {{ $value.bearerToken }}" \ + {{- end }} + {{- if $value.basic }} + -H "Authorization: Basic {{ $value.basic }}" \ + {{- end }} + {{- if $value.gitlabToken }} + -H "PRIVATE-TOKEN: {{ $value.gitlabToken }}" \ + {{- end }} + -H "Content-Type: application/json;charset=UTF-8" \ + {{- end }} + {{- $dpPath := "" -}} + {{- range $kd := (index $dashboardProviders "dashboardproviders.yaml").providers }} + {{- if eq $kd.name $provider }} + {{- $dpPath = $kd.options.path }} + {{- end }} + {{- end }} + {{- if $value.url }} + "{{ $value.url }}" \ + {{- else }} + "https://grafana.com/api/dashboards/{{ $value.gnetId }}/revisions/{{- if $value.revision -}}{{ $value.revision }}{{- else -}}1{{- end -}}/download" \ + {{- end }} + {{- if $value.datasource }} + {{- if kindIs "string" $value.datasource }} + | sed '/-- .* --/! s/"datasource":.*,/"datasource": "{{ $value.datasource }}",/g' \ + {{- end }} + {{- if kindIs "slice" $value.datasource }} + {{- range $value.datasource }} + | sed '/-- .* --/! s/${{"{"}}{{ .name }}}/{{ .value }}/g' \ + {{- end }} + {{- end }} + {{- end }} + {{- if $value.b64content }} + | base64 -d \ + {{- end }} + > "{{- if $dpPath -}}{{ $dpPath }}{{- else -}}/var/lib/grafana/dashboards/{{ $provider }}{{- end -}}/{{ $key }}.json" + {{ end }} + {{- end }} +{{- end }} +{{- end }} +{{- end -}} + +{{/* + Generate dashboard json config map data + */}} +{{- define "grafana.configDashboardProviderData" -}} +provider.yaml: |- + apiVersion: 1 + providers: + - name: '{{ .Values.sidecar.dashboards.provider.name }}' + orgId: {{ .Values.sidecar.dashboards.provider.orgid }} + {{- if not .Values.sidecar.dashboards.provider.foldersFromFilesStructure }} + folder: '{{ .Values.sidecar.dashboards.provider.folder }}' + folderUid: '{{ .Values.sidecar.dashboards.provider.folderUid }}' + {{- end }} + type: {{ .Values.sidecar.dashboards.provider.type }} + disableDeletion: {{ .Values.sidecar.dashboards.provider.disableDelete }} + allowUiUpdates: {{ .Values.sidecar.dashboards.provider.allowUiUpdates }} + updateIntervalSeconds: {{ .Values.sidecar.dashboards.provider.updateIntervalSeconds | default 30 }} + options: + foldersFromFilesStructure: {{ .Values.sidecar.dashboards.provider.foldersFromFilesStructure }} + path: {{ .Values.sidecar.dashboards.folder }}{{- with .Values.sidecar.dashboards.defaultFolderName }}/{{ . }}{{- end }} +{{- end -}} + +{{- define "grafana.secretsData" -}} +{{- if and (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) }} +admin-user: {{ .Values.adminUser | b64enc | quote }} +{{- if .Values.adminPassword }} +admin-password: {{ .Values.adminPassword | b64enc | quote }} +{{- else }} +admin-password: {{ include "grafana.password" . }} +{{- end }} +{{- end }} +{{- if not .Values.ldap.existingSecret }} +ldap-toml: {{ tpl .Values.ldap.config $ | b64enc | quote }} +{{- end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_helpers.tpl new file mode 100644 index 000000000..2a68cb6f8 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_helpers.tpl @@ -0,0 +1,276 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "grafana.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 "grafana.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 "grafana.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Create the name of the service account +*/}} +{{- define "grafana.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "grafana.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{- define "grafana.serviceAccountNameTest" -}} +{{- if .Values.serviceAccount.create }} +{{- default (print (include "grafana.fullname" .) "-test") .Values.serviceAccount.nameTest }} +{{- else }} +{{- default "default" .Values.serviceAccount.nameTest }} +{{- end }} +{{- end }} + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts +*/}} +{{- define "grafana.namespace" -}} +{{- if .Values.namespaceOverride }} +{{- .Values.namespaceOverride }} +{{- else }} +{{- .Release.Namespace }} +{{- end }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "grafana.labels" -}} +helm.sh/chart: {{ include "grafana.chart" . }} +{{ include "grafana.selectorLabels" . }} +{{- if or .Chart.AppVersion .Values.image.tag }} +app.kubernetes.io/version: {{ mustRegexReplaceAllLiteral "@sha.*" .Values.image.tag "" | default .Chart.AppVersion | trunc 63 | trimSuffix "-" | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- with .Values.extraLabels }} +{{ toYaml . }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "grafana.selectorLabels" -}} +app.kubernetes.io/name: {{ include "grafana.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "grafana.imageRenderer.labels" -}} +helm.sh/chart: {{ include "grafana.chart" . }} +{{ include "grafana.imageRenderer.selectorLabels" . }} +{{- if or .Chart.AppVersion .Values.image.tag }} +app.kubernetes.io/version: {{ mustRegexReplaceAllLiteral "@sha.*" .Values.image.tag "" | default .Chart.AppVersion | trunc 63 | trimSuffix "-" | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels ImageRenderer +*/}} +{{- define "grafana.imageRenderer.selectorLabels" -}} +app.kubernetes.io/name: {{ include "grafana.name" . }}-image-renderer +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Looks if there's an existing secret and reuse its password. If not it generates +new password and use it. +*/}} +{{- define "grafana.password" -}} +{{- $secret := (lookup "v1" "Secret" (include "grafana.namespace" .) (include "grafana.fullname" .) ) }} +{{- if $secret }} +{{- index $secret "data" "admin-password" }} +{{- else }} +{{- (randAlphaNum 40) | b64enc | quote }} +{{- end }} +{{- end }} + +{{/* +Return the appropriate apiVersion for rbac. +*/}} +{{- define "grafana.rbac.apiVersion" -}} +{{- if $.Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1" }} +{{- print "rbac.authorization.k8s.io/v1" }} +{{- else }} +{{- print "rbac.authorization.k8s.io/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "grafana.ingress.apiVersion" -}} +{{- if and ($.Capabilities.APIVersions.Has "networking.k8s.io/v1") (semverCompare ">= 1.19-0" .Capabilities.KubeVersion.Version) }} +{{- print "networking.k8s.io/v1" }} +{{- else if $.Capabilities.APIVersions.Has "networking.k8s.io/v1beta1" }} +{{- print "networking.k8s.io/v1beta1" }} +{{- else }} +{{- print "extensions/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Return the appropriate apiVersion for Horizontal Pod Autoscaler. +*/}} +{{- define "grafana.hpa.apiVersion" -}} +{{- if .Capabilities.APIVersions.Has "autoscaling/v2" }} +{{- print "autoscaling/v2" }} +{{- else }} +{{- print "autoscaling/v2beta2" }} +{{- end }} +{{- end }} + +{{/* +Return the appropriate apiVersion for podDisruptionBudget. +*/}} +{{- define "grafana.podDisruptionBudget.apiVersion" -}} +{{- if $.Values.podDisruptionBudget.apiVersion }} +{{- print $.Values.podDisruptionBudget.apiVersion }} +{{- else if $.Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} +{{- print "policy/v1" }} +{{- else }} +{{- print "policy/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Return if ingress is stable. +*/}} +{{- define "grafana.ingress.isStable" -}} +{{- eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1" }} +{{- end }} + +{{/* +Return if ingress supports ingressClassName. +*/}} +{{- define "grafana.ingress.supportsIngressClassName" -}} +{{- or (eq (include "grafana.ingress.isStable" .) "true") (and (eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18-0" .Capabilities.KubeVersion.Version)) }} +{{- end }} + +{{/* +Return if ingress supports pathType. +*/}} +{{- define "grafana.ingress.supportsPathType" -}} +{{- or (eq (include "grafana.ingress.isStable" .) "true") (and (eq (include "grafana.ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18-0" .Capabilities.KubeVersion.Version)) }} +{{- end }} + +{{/* +Formats imagePullSecrets. Input is (dict "root" . "imagePullSecrets" .{specific imagePullSecrets}) +*/}} +{{- define "grafana.imagePullSecrets" -}} +{{- $root := .root }} +{{- range (concat .root.Values.global.imagePullSecrets .imagePullSecrets) }} +{{- if eq (typeOf .) "map[string]interface {}" }} +- {{ toYaml (dict "name" (tpl .name $root)) | trim }} +{{- else }} +- name: {{ tpl . $root }} +{{- end }} +{{- end }} +{{- end }} + + +{{/* + Checks whether or not the configSecret secret has to be created + */}} +{{- define "grafana.shouldCreateConfigSecret" -}} +{{- $secretFound := false -}} +{{- range $key, $value := .Values.datasources }} + {{- if hasKey $value "secret" }} + {{- $secretFound = true}} + {{- end }} +{{- end }} +{{- range $key, $value := .Values.notifiers }} + {{- if hasKey $value "secret" }} + {{- $secretFound = true}} + {{- end }} +{{- end }} +{{- range $key, $value := .Values.alerting }} + {{- if (or (hasKey $value "secret") (hasKey $value "secretFile")) }} + {{- $secretFound = true}} + {{- end }} +{{- end }} +{{- $secretFound}} +{{- end -}} + +{{/* + Checks whether the user is attempting to store secrets in plaintext + in the grafana.ini configmap +*/}} +{{/* grafana.assertNoLeakedSecrets checks for sensitive keys in values */}} +{{- define "grafana.assertNoLeakedSecrets" -}} + {{- $sensitiveKeysYaml := ` +sensitiveKeys: +- path: ["database", "password"] +- path: ["smtp", "password"] +- path: ["security", "secret_key"] +- path: ["security", "admin_password"] +- path: ["auth.basic", "password"] +- path: ["auth.ldap", "bind_password"] +- path: ["auth.google", "client_secret"] +- path: ["auth.github", "client_secret"] +- path: ["auth.gitlab", "client_secret"] +- path: ["auth.generic_oauth", "client_secret"] +- path: ["auth.okta", "client_secret"] +- path: ["auth.azuread", "client_secret"] +- path: ["auth.grafana_com", "client_secret"] +- path: ["auth.grafananet", "client_secret"] +- path: ["azure", "user_identity_client_secret"] +- path: ["unified_alerting", "ha_redis_password"] +- path: ["metrics", "basic_auth_password"] +- path: ["external_image_storage.s3", "secret_key"] +- path: ["external_image_storage.webdav", "password"] +- path: ["external_image_storage.azure_blob", "account_key"] +` | fromYaml -}} + {{- if $.Values.assertNoLeakedSecrets -}} + {{- $grafanaIni := index .Values "grafana.ini" -}} + {{- range $_, $secret := $sensitiveKeysYaml.sensitiveKeys -}} + {{- $currentMap := $grafanaIni -}} + {{- $shouldContinue := true -}} + {{- range $index, $elem := $secret.path -}} + {{- if and $shouldContinue (hasKey $currentMap $elem) -}} + {{- if eq (len $secret.path) (add1 $index) -}} + {{- if not (regexMatch "\\$(?:__(?:env|file|vault))?{[^}]+}" (index $currentMap $elem)) -}} + {{- fail (printf "Sensitive key '%s' should not be defined explicitly in values. Use variable expansion instead. You can disable this client-side validation by changing the value of assertNoLeakedSecrets." (join "." $secret.path)) -}} + {{- end -}} + {{- else -}} + {{- $currentMap = index $currentMap $elem -}} + {{- end -}} + {{- else -}} + {{- $shouldContinue = false -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/_pod.tpl b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_pod.tpl new file mode 100644 index 000000000..a8b104b5d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/_pod.tpl @@ -0,0 +1,1329 @@ +{{- define "grafana.pod" -}} +{{- $sts := list "sts" "StatefulSet" "statefulset" -}} +{{- $root := . -}} +{{- with .Values.schedulerName }} +schedulerName: "{{ . }}" +{{- end }} +serviceAccountName: {{ include "grafana.serviceAccountName" . }} +automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} +{{- with .Values.securityContext }} +securityContext: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.hostAliases }} +hostAliases: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- if .Values.dnsPolicy }} +dnsPolicy: {{ .Values.dnsPolicy }} +{{- end }} +{{- with .Values.dnsConfig }} +dnsConfig: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.priorityClassName }} +priorityClassName: {{ . }} +{{- end }} +{{- if ( or .Values.persistence.enabled .Values.dashboards .Values.extraInitContainers (and .Values.sidecar.alerts.enabled .Values.sidecar.alerts.initAlerts) (and .Values.sidecar.datasources.enabled .Values.sidecar.datasources.initDatasources) (and .Values.sidecar.notifiers.enabled .Values.sidecar.notifiers.initNotifiers)) }} +initContainers: +{{- end }} +{{- if ( and .Values.persistence.enabled .Values.initChownData.enabled ) }} + - name: init-chown-data + {{- $registry := .Values.global.imageRegistry | default .Values.initChownData.image.registry -}} + {{- if .Values.initChownData.image.sha }} + image: "{{ $registry }}/{{ .Values.initChownData.image.repository }}{{ if .Values.initChownData.image.tag }}:{{ .Values.initChownData.image.tag }}{{ end }}@sha256:{{ .Values.initChownData.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.initChownData.image.repository }}{{ if .Values.initChownData.image.tag }}:{{ .Values.initChownData.image.tag }}{{ end }}" + {{- end }} + imagePullPolicy: {{ .Values.initChownData.image.pullPolicy }} + {{- with .Values.initChownData.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + command: + - chown + - -R + - {{ .Values.securityContext.runAsUser }}:{{ .Values.securityContext.runAsGroup }} + - /var/lib/grafana + {{- with .Values.initChownData.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: storage + mountPath: "/var/lib/grafana" + {{- with .Values.persistence.subPath }} + subPath: {{ tpl . $root }} + {{- end }} +{{- end }} +{{- if .Values.dashboards }} + - name: download-dashboards + {{- $registry := .Values.global.imageRegistry | default .Values.downloadDashboardsImage.registry -}} + {{- if .Values.downloadDashboardsImage.sha }} + image: "{{ $registry }}/{{ .Values.downloadDashboardsImage.repository }}{{ if .Values.downloadDashboardsImage.tag }}:{{ .Values.downloadDashboardsImage.tag }}{{ end }}@sha256:{{ .Values.downloadDashboardsImage.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.downloadDashboardsImage.repository }}{{ if .Values.downloadDashboardsImage.tag }}:{{ .Values.downloadDashboardsImage.tag }}{{ end }}" + {{- end }} + imagePullPolicy: {{ .Values.downloadDashboardsImage.pullPolicy }} + command: ["/bin/sh"] + args: [ "-c", "mkdir -p /var/lib/grafana/dashboards/default && /bin/sh -x /etc/grafana/download_dashboards.sh" ] + {{- with .Values.downloadDashboards.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + env: + {{- range $key, $value := .Values.downloadDashboards.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- range $key, $value := .Values.downloadDashboards.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 10 }} + {{- end }} + {{- with .Values.downloadDashboards.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.downloadDashboards.envFromSecret }} + envFrom: + - secretRef: + name: {{ tpl . $root }} + {{- end }} + volumeMounts: + - name: config + mountPath: "/etc/grafana/download_dashboards.sh" + subPath: download_dashboards.sh + - name: storage + mountPath: "/var/lib/grafana" + {{- with .Values.persistence.subPath }} + subPath: {{ tpl . $root }} + {{- end }} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + {{- end }} +{{- end }} +{{- if and .Values.sidecar.alerts.enabled .Values.sidecar.alerts.initAlerts }} + - name: {{ include "grafana.name" . }}-init-sc-alerts + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.alerts.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- if .Values.sidecar.alerts.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: "LIST" + - name: LABEL + value: "{{ .Values.sidecar.alerts.label }}" + {{- with .Values.sidecar.alerts.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.alerts.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.alerts.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/alerting" + - name: RESOURCE + value: {{ quote .Values.sidecar.alerts.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.alerts.searchNamespace }} + - name: NAMESPACE + value: {{ . | join "," | quote }} + {{- end }} + {{- with .Values.sidecar.alerts.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: {{ quote . }} + {{- end }} + {{- with .Values.sidecar.alerts.script }} + - name: SCRIPT + value: {{ quote . }} + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-alerts-volume + mountPath: "/etc/grafana/provisioning/alerting" + {{- with .Values.sidecar.alerts.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end }} +{{- if and .Values.sidecar.datasources.enabled .Values.sidecar.datasources.initDatasources }} + - name: {{ include "grafana.name" . }}-init-sc-datasources + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.datasources.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- range $key, $value := .Values.sidecar.datasources.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 10 }} + {{- end }} + {{- if .Values.sidecar.datasources.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: "LIST" + - name: LABEL + value: "{{ .Values.sidecar.datasources.label }}" + {{- with .Values.sidecar.datasources.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.datasources.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.datasources.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/datasources" + - name: RESOURCE + value: {{ quote .Values.sidecar.datasources.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- if .Values.sidecar.datasources.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (.Values.sidecar.datasources.searchNamespace | join ",") . }}" + {{- end }} + {{- with .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" +{{- end }} +{{- if and .Values.sidecar.notifiers.enabled .Values.sidecar.notifiers.initNotifiers }} + - name: {{ include "grafana.name" . }}-init-sc-notifiers + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.notifiers.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- if .Values.sidecar.notifiers.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: LIST + - name: LABEL + value: "{{ .Values.sidecar.notifiers.label }}" + {{- with .Values.sidecar.notifiers.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.notifiers.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.notifiers.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/notifiers" + - name: RESOURCE + value: {{ quote .Values.sidecar.notifiers.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.notifiers.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (. | join ",") $root }}" + {{- end }} + {{- with .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-notifiers-volume + mountPath: "/etc/grafana/provisioning/notifiers" +{{- end}} +{{- with .Values.extraInitContainers }} + {{- tpl (toYaml .) $root | nindent 2 }} +{{- end }} +{{- if or .Values.image.pullSecrets .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- include "grafana.imagePullSecrets" (dict "root" $root "imagePullSecrets" .Values.image.pullSecrets) | nindent 2 }} +{{- end }} +{{- if not .Values.enableKubeBackwardCompatibility }} +enableServiceLinks: {{ .Values.enableServiceLinks }} +{{- end }} +containers: +{{- if and .Values.sidecar.alerts.enabled (not .Values.sidecar.alerts.initAlerts) }} + - name: {{ include "grafana.name" . }}-sc-alerts + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.alerts.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- if .Values.sidecar.alerts.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: {{ .Values.sidecar.alerts.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.alerts.label }}" + {{- with .Values.sidecar.alerts.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.alerts.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.alerts.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/alerting" + - name: RESOURCE + value: {{ quote .Values.sidecar.alerts.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.alerts.searchNamespace }} + - name: NAMESPACE + value: {{ . | join "," | quote }} + {{- end }} + {{- with .Values.sidecar.alerts.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: {{ quote . }} + {{- end }} + {{- with .Values.sidecar.alerts.script }} + - name: SCRIPT + value: {{ quote . }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.alerts.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.alerts.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} + {{- if .Values.sidecar.alerts.watchServerTimeout }} + {{- if ne .Values.sidecar.alerts.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.alerts.watchServerTimeout with .Values.sidecar.alerts.watchMethod %s" .Values.sidecar.alerts.watchMethod) }} + {{- end }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.alerts.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.alerts.watchClientTimeout }} + {{- if ne .Values.sidecar.alerts.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.alerts.watchClientTimeout with .Values.sidecar.alerts.watchMethod %s" .Values.sidecar.alerts.watchMethod) }} + {{- end }} + - name: WATCH_CLIENT_TIMEOUT + value: "{{ .Values.sidecar.alerts.watchClientTimeout }}" + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-alerts-volume + mountPath: "/etc/grafana/provisioning/alerting" + {{- with .Values.sidecar.alerts.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end}} +{{- if .Values.sidecar.dashboards.enabled }} + - name: {{ include "grafana.name" . }}-sc-dashboard + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.dashboards.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- range $key, $value := .Values.sidecar.dashboards.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 10 }} + {{- end }} + {{- if .Values.sidecar.dashboards.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: {{ .Values.sidecar.dashboards.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.dashboards.label }}" + {{- with .Values.sidecar.dashboards.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.dashboards.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.dashboards.logLevel }} + {{- end }} + - name: FOLDER + value: "{{ .Values.sidecar.dashboards.folder }}{{- with .Values.sidecar.dashboards.defaultFolderName }}/{{ . }}{{- end }}" + - name: RESOURCE + value: {{ quote .Values.sidecar.dashboards.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.dashboards.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (. | join ",") $root }}" + {{- end }} + {{- with .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.dashboards.folderAnnotation }} + - name: FOLDER_ANNOTATION + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.dashboards.script }} + - name: SCRIPT + value: "{{ . }}" + {{- end }} + {{- if not .Values.sidecar.dashboards.skipReload }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + - name: REQ_URL + value: {{ .Values.sidecar.dashboards.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} + {{- if .Values.sidecar.dashboards.watchServerTimeout }} + {{- if ne .Values.sidecar.dashboards.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.dashboards.watchServerTimeout with .Values.sidecar.dashboards.watchMethod %s" .Values.sidecar.dashboards.watchMethod) }} + {{- end }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.dashboards.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.dashboards.watchClientTimeout }} + {{- if ne .Values.sidecar.dashboards.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.dashboards.watchClientTimeout with .Values.sidecar.dashboards.watchMethod %s" .Values.sidecar.dashboards.watchMethod) }} + {{- end }} + - name: WATCH_CLIENT_TIMEOUT + value: {{ .Values.sidecar.dashboards.watchClientTimeout | quote }} + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-dashboard-volume + mountPath: {{ .Values.sidecar.dashboards.folder | quote }} + {{- with .Values.sidecar.dashboards.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end}} +{{- if and .Values.sidecar.datasources.enabled (not .Values.sidecar.datasources.initDatasources) }} + - name: {{ include "grafana.name" . }}-sc-datasources + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.datasources.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- range $key, $value := .Values.sidecar.datasources.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 10 }} + {{- end }} + {{- if .Values.sidecar.datasources.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: {{ .Values.sidecar.datasources.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.datasources.label }}" + {{- with .Values.sidecar.datasources.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.datasources.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.datasources.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/datasources" + - name: RESOURCE + value: {{ quote .Values.sidecar.datasources.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.datasources.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (. | join ",") $root }}" + {{- end }} + {{- if .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ .Values.sidecar.skipTlsVerify }}" + {{- end }} + {{- if .Values.sidecar.datasources.script }} + - name: SCRIPT + value: "{{ .Values.sidecar.datasources.script }}" + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.datasources.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.datasources.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} + {{- if .Values.sidecar.datasources.watchServerTimeout }} + {{- if ne .Values.sidecar.datasources.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.datasources.watchServerTimeout with .Values.sidecar.datasources.watchMethod %s" .Values.sidecar.datasources.watchMethod) }} + {{- end }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.datasources.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.datasources.watchClientTimeout }} + {{- if ne .Values.sidecar.datasources.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.datasources.watchClientTimeout with .Values.sidecar.datasources.watchMethod %s" .Values.sidecar.datasources.watchMethod) }} + {{- end }} + - name: WATCH_CLIENT_TIMEOUT + value: "{{ .Values.sidecar.datasources.watchClientTimeout }}" + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" + {{- with .Values.sidecar.datasources.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end}} +{{- if .Values.sidecar.notifiers.enabled }} + - name: {{ include "grafana.name" . }}-sc-notifiers + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.notifiers.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- if .Values.sidecar.notifiers.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: {{ .Values.sidecar.notifiers.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.notifiers.label }}" + {{- with .Values.sidecar.notifiers.labelValue }} + - name: LABEL_VALUE + value: {{ quote . }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.notifiers.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.notifiers.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/notifiers" + - name: RESOURCE + value: {{ quote .Values.sidecar.notifiers.resource }} + {{- if .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ .Values.sidecar.enableUniqueFilenames }}" + {{- end }} + {{- with .Values.sidecar.notifiers.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (. | join ",") $root }}" + {{- end }} + {{- with .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ . }}" + {{- end }} + {{- if .Values.sidecar.notifiers.script }} + - name: SCRIPT + value: "{{ .Values.sidecar.notifiers.script }}" + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.notifiers.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.notifiers.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} + {{- if .Values.sidecar.notifiers.watchServerTimeout }} + {{- if ne .Values.sidecar.notifiers.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.notifiers.watchServerTimeout with .Values.sidecar.notifiers.watchMethod %s" .Values.sidecar.notifiers.watchMethod) }} + {{- end }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.notifiers.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.notifiers.watchClientTimeout }} + {{- if ne .Values.sidecar.notifiers.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.notifiers.watchClientTimeout with .Values.sidecar.notifiers.watchMethod %s" .Values.sidecar.notifiers.watchMethod) }} + {{- end }} + - name: WATCH_CLIENT_TIMEOUT + value: "{{ .Values.sidecar.notifiers.watchClientTimeout }}" + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-notifiers-volume + mountPath: "/etc/grafana/provisioning/notifiers" + {{- with .Values.sidecar.notifiers.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end}} +{{- if .Values.sidecar.plugins.enabled }} + - name: {{ include "grafana.name" . }}-sc-plugins + {{- $registry := .Values.global.imageRegistry | default .Values.sidecar.image.registry -}} + {{- if .Values.sidecar.image.sha }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}@sha256:{{ .Values.sidecar.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.sidecar.image.repository }}:{{ .Values.sidecar.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.sidecar.imagePullPolicy }} + env: + {{- range $key, $value := .Values.sidecar.plugins.env }} + - name: "{{ $key }}" + value: "{{ $value }}" + {{- end }} + {{- if .Values.sidecar.plugins.ignoreAlreadyProcessed }} + - name: IGNORE_ALREADY_PROCESSED + value: "true" + {{- end }} + - name: METHOD + value: {{ .Values.sidecar.plugins.watchMethod }} + - name: LABEL + value: "{{ .Values.sidecar.plugins.label }}" + {{- if .Values.sidecar.plugins.labelValue }} + - name: LABEL_VALUE + value: {{ quote .Values.sidecar.plugins.labelValue }} + {{- end }} + {{- if or .Values.sidecar.logLevel .Values.sidecar.plugins.logLevel }} + - name: LOG_LEVEL + value: {{ default .Values.sidecar.logLevel .Values.sidecar.plugins.logLevel }} + {{- end }} + - name: FOLDER + value: "/etc/grafana/provisioning/plugins" + - name: RESOURCE + value: {{ quote .Values.sidecar.plugins.resource }} + {{- with .Values.sidecar.enableUniqueFilenames }} + - name: UNIQUE_FILENAMES + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.plugins.searchNamespace }} + - name: NAMESPACE + value: "{{ tpl (. | join ",") $root }}" + {{- end }} + {{- with .Values.sidecar.plugins.script }} + - name: SCRIPT + value: "{{ . }}" + {{- end }} + {{- with .Values.sidecar.skipTlsVerify }} + - name: SKIP_TLS_VERIFY + value: "{{ . }}" + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_USERNAME + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: REQ_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if not .Values.sidecar.plugins.skipReload }} + - name: REQ_URL + value: {{ .Values.sidecar.plugins.reloadURL }} + - name: REQ_METHOD + value: POST + {{- end }} + {{- if .Values.sidecar.plugins.watchServerTimeout }} + {{- if ne .Values.sidecar.plugins.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.plugins.watchServerTimeout with .Values.sidecar.plugins.watchMethod %s" .Values.sidecar.plugins.watchMethod) }} + {{- end }} + - name: WATCH_SERVER_TIMEOUT + value: "{{ .Values.sidecar.plugins.watchServerTimeout }}" + {{- end }} + {{- if .Values.sidecar.plugins.watchClientTimeout }} + {{- if ne .Values.sidecar.plugins.watchMethod "WATCH" }} + {{- fail (printf "Cannot use .Values.sidecar.plugins.watchClientTimeout with .Values.sidecar.plugins.watchMethod %s" .Values.sidecar.plugins.watchMethod) }} + {{- end }} + - name: WATCH_CLIENT_TIMEOUT + value: "{{ .Values.sidecar.plugins.watchClientTimeout }}" + {{- end }} + {{- with .Values.sidecar.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.sidecar.securityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: sc-plugins-volume + mountPath: "/etc/grafana/provisioning/plugins" + {{- with .Values.sidecar.plugins.extraMounts }} + {{- toYaml . | trim | nindent 6 }} + {{- end }} +{{- end}} + - name: {{ .Chart.Name }} + {{- $registry := .Values.global.imageRegistry | default .Values.image.registry -}} + {{- if .Values.image.sha }} + image: "{{ $registry }}/{{ .Values.image.repository }}{{ if .Values.image.tag }}:{{ .Values.image.tag | default .Chart.AppVersion }}{{ end }}@sha256:{{ .Values.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.image.repository }}{{ if .Values.image.tag }}:{{ .Values.image.tag | default .Chart.AppVersion }}{{ end }}" + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + {{- if .Values.command }} + command: + {{- range .Values.command }} + - {{ . | quote }} + {{- end }} + {{- end }} + {{- if .Values.args }} + args: + {{- range .Values.args }} + - {{ . | quote }} + {{- end }} + {{- end }} + {{- with .Values.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: config + mountPath: "/etc/grafana/grafana.ini" + subPath: grafana.ini + {{- if .Values.ldap.enabled }} + - name: ldap + mountPath: "/etc/grafana/ldap.toml" + subPath: ldap.toml + {{- end }} + {{- range .Values.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + mountPath: {{ tpl .mountPath $root }} + subPath: {{ tpl (.subPath | default "") $root }} + readOnly: {{ .readOnly }} + {{- end }} + - name: storage + mountPath: "/var/lib/grafana" + {{- with .Values.persistence.subPath }} + subPath: {{ tpl . $root }} + {{- end }} + {{- with .Values.dashboards }} + {{- range $provider, $dashboards := . }} + {{- range $key, $value := $dashboards }} + {{- if (or (hasKey $value "json") (hasKey $value "file")) }} + - name: dashboards-{{ $provider }} + mountPath: "/var/lib/grafana/dashboards/{{ $provider }}/{{ $key }}.json" + subPath: "{{ $key }}.json" + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- with .Values.dashboardsConfigMaps }} + {{- range (keys . | sortAlpha) }} + - name: dashboards-{{ . }} + mountPath: "/var/lib/grafana/dashboards/{{ . }}" + {{- end }} + {{- end }} + {{- with .Values.datasources }} + {{- $datasources := . }} + {{- range (keys . | sortAlpha) }} + {{- if (or (hasKey (index $datasources .) "secret")) }} {{/*check if current datasource should be handeled as secret */}} + - name: config-secret + mountPath: "/etc/grafana/provisioning/datasources/{{ . }}" + subPath: {{ . | quote }} + {{- else }} + - name: config + mountPath: "/etc/grafana/provisioning/datasources/{{ . }}" + subPath: {{ . | quote }} + {{- end }} + {{- end }} + {{- end }} + {{- with .Values.notifiers }} + {{- $notifiers := . }} + {{- range (keys . | sortAlpha) }} + {{- if (or (hasKey (index $notifiers .) "secret")) }} {{/*check if current notifier should be handeled as secret */}} + - name: config-secret + mountPath: "/etc/grafana/provisioning/notifiers/{{ . }}" + subPath: {{ . | quote }} + {{- else }} + - name: config + mountPath: "/etc/grafana/provisioning/notifiers/{{ . }}" + subPath: {{ . | quote }} + {{- end }} + {{- end }} + {{- end }} + {{- with .Values.alerting }} + {{- $alertingmap := .}} + {{- range (keys . | sortAlpha) }} + {{- if (or (hasKey (index $.Values.alerting .) "secret") (hasKey (index $.Values.alerting .) "secretFile")) }} {{/*check if current alerting entry should be handeled as secret */}} + - name: config-secret + mountPath: "/etc/grafana/provisioning/alerting/{{ . }}" + subPath: {{ . | quote }} + {{- else }} + - name: config + mountPath: "/etc/grafana/provisioning/alerting/{{ . }}" + subPath: {{ . | quote }} + {{- end }} + {{- end }} + {{- end }} + {{- with .Values.dashboardProviders }} + {{- range (keys . | sortAlpha) }} + - name: config + mountPath: "/etc/grafana/provisioning/dashboards/{{ . }}" + subPath: {{ . | quote }} + {{- end }} + {{- end }} + {{- with .Values.sidecar.alerts.enabled }} + - name: sc-alerts-volume + mountPath: "/etc/grafana/provisioning/alerting" + {{- end}} + {{- if .Values.sidecar.dashboards.enabled }} + - name: sc-dashboard-volume + mountPath: {{ .Values.sidecar.dashboards.folder | quote }} + {{- if .Values.sidecar.dashboards.SCProvider }} + - name: sc-dashboard-provider + mountPath: "/etc/grafana/provisioning/dashboards/sc-dashboardproviders.yaml" + subPath: provider.yaml + {{- end}} + {{- end}} + {{- if .Values.sidecar.datasources.enabled }} + - name: sc-datasources-volume + mountPath: "/etc/grafana/provisioning/datasources" + {{- end}} + {{- if .Values.sidecar.plugins.enabled }} + - name: sc-plugins-volume + mountPath: "/etc/grafana/provisioning/plugins" + {{- end}} + {{- if .Values.sidecar.notifiers.enabled }} + - name: sc-notifiers-volume + mountPath: "/etc/grafana/provisioning/notifiers" + {{- end}} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + subPath: {{ .subPath | default "" }} + {{- end }} + {{- range .Values.extraVolumeMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath | default "" }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.extraEmptyDirMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + {{- end }} + ports: + - name: {{ .Values.podPortName }} + containerPort: {{ .Values.service.targetPort }} + protocol: TCP + - name: {{ .Values.gossipPortName }}-tcp + containerPort: 9094 + protocol: TCP + - name: {{ .Values.gossipPortName }}-udp + containerPort: 9094 + protocol: UDP + env: + - name: POD_IP + valueFrom: + fieldRef: + fieldPath: status.podIP + {{- if and (not .Values.env.GF_SECURITY_ADMIN_USER) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: GF_SECURITY_ADMIN_USER + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.userKey | default "admin-user" }} + {{- end }} + {{- if and (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + - name: GF_SECURITY_ADMIN_PASSWORD + valueFrom: + secretKeyRef: + name: {{ (tpl .Values.admin.existingSecret .) | default (include "grafana.fullname" .) }} + key: {{ .Values.admin.passwordKey | default "admin-password" }} + {{- end }} + {{- if .Values.plugins }} + - name: GF_INSTALL_PLUGINS + valueFrom: + configMapKeyRef: + name: {{ include "grafana.fullname" . }} + key: plugins + {{- end }} + {{- if .Values.smtp.existingSecret }} + - name: GF_SMTP_USER + valueFrom: + secretKeyRef: + name: {{ .Values.smtp.existingSecret }} + key: {{ .Values.smtp.userKey | default "user" }} + - name: GF_SMTP_PASSWORD + valueFrom: + secretKeyRef: + name: {{ .Values.smtp.existingSecret }} + key: {{ .Values.smtp.passwordKey | default "password" }} + {{- end }} + {{- if .Values.imageRenderer.enabled }} + - name: GF_RENDERING_SERVER_URL + {{- if .Values.imageRenderer.serverURL }} + value: {{ .Values.imageRenderer.serverURL | quote }} + {{- else }} + value: http://{{ include "grafana.fullname" . }}-image-renderer.{{ include "grafana.namespace" . }}:{{ .Values.imageRenderer.service.port }}/render + {{- end }} + - name: GF_RENDERING_CALLBACK_URL + {{- if .Values.imageRenderer.renderingCallbackURL }} + value: {{ .Values.imageRenderer.renderingCallbackURL | quote }} + {{- else }} + value: {{ .Values.imageRenderer.grafanaProtocol }}://{{ include "grafana.fullname" . }}.{{ include "grafana.namespace" . }}:{{ .Values.service.port }}/{{ .Values.imageRenderer.grafanaSubPath }} + {{- end }} + {{- end }} + - name: GF_PATHS_DATA + value: {{ (get .Values "grafana.ini").paths.data }} + - name: GF_PATHS_LOGS + value: {{ (get .Values "grafana.ini").paths.logs }} + - name: GF_PATHS_PLUGINS + value: {{ (get .Values "grafana.ini").paths.plugins }} + - name: GF_PATHS_PROVISIONING + value: {{ (get .Values "grafana.ini").paths.provisioning }} + {{- range $key, $value := .Values.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 10 }} + {{- end }} + {{- range $key, $value := .Values.env }} + - name: "{{ tpl $key $ }}" + value: "{{ tpl (print $value) $ }}" + {{- end }} + {{- if or .Values.envFromSecret (or .Values.envRenderSecret .Values.envFromSecrets) .Values.envFromConfigMaps }} + envFrom: + {{- if .Values.envFromSecret }} + - secretRef: + name: {{ tpl .Values.envFromSecret . }} + {{- end }} + {{- if .Values.envRenderSecret }} + - secretRef: + name: {{ include "grafana.fullname" . }}-env + {{- end }} + {{- range .Values.envFromSecrets }} + - secretRef: + name: {{ tpl .name $ }} + optional: {{ .optional | default false }} + {{- if .prefix }} + prefix: {{ tpl .prefix $ }} + {{- end }} + {{- end }} + {{- range .Values.envFromConfigMaps }} + - configMapRef: + name: {{ tpl .name $ }} + optional: {{ .optional | default false }} + {{- if .prefix }} + prefix: {{ tpl .prefix $ }} + {{- end }} + {{- end }} + {{- end }} + {{- with .Values.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.lifecycleHooks }} + lifecycle: + {{- tpl (toYaml .) $root | nindent 6 }} + {{- end }} + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} +{{- with .Values.extraContainers }} + {{- tpl . $ | nindent 2 }} +{{- end }} +{{- with .Values.nodeSelector }} +nodeSelector: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.affinity }} +affinity: + {{- tpl (toYaml .) $root | nindent 2 }} +{{- end }} +{{- with .Values.topologySpreadConstraints }} +topologySpreadConstraints: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.tolerations }} +tolerations: + {{- toYaml . | nindent 2 }} +{{- end }} +volumes: + - name: config + configMap: + name: {{ include "grafana.fullname" . }} + {{- $createConfigSecret := eq (include "grafana.shouldCreateConfigSecret" .) "true" -}} + {{- if and .Values.createConfigmap $createConfigSecret }} + - name: config-secret + secret: + secretName: {{ include "grafana.fullname" . }}-config-secret + {{- end }} + {{- range .Values.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + configMap: + name: {{ tpl .configMap $root }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- with .items }} + items: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- end }} + {{- if .Values.dashboards }} + {{- range (keys .Values.dashboards | sortAlpha) }} + - name: dashboards-{{ . }} + configMap: + name: {{ include "grafana.fullname" $ }}-dashboards-{{ . }} + {{- end }} + {{- end }} + {{- if .Values.dashboardsConfigMaps }} + {{- range $provider, $name := .Values.dashboardsConfigMaps }} + - name: dashboards-{{ $provider }} + configMap: + name: {{ tpl $name $root }} + {{- end }} + {{- end }} + {{- if .Values.ldap.enabled }} + - name: ldap + secret: + {{- if .Values.ldap.existingSecret }} + secretName: {{ .Values.ldap.existingSecret }} + {{- else }} + secretName: {{ include "grafana.fullname" . }} + {{- end }} + items: + - key: ldap-toml + path: ldap.toml + {{- end }} + {{- if and .Values.persistence.enabled (eq .Values.persistence.type "pvc") }} + - name: storage + persistentVolumeClaim: + claimName: {{ tpl (.Values.persistence.existingClaim | default (include "grafana.fullname" .)) . }} + {{- else if and .Values.persistence.enabled (has .Values.persistence.type $sts) }} + {{/* nothing */}} + {{- else }} + - name: storage + {{- if .Values.persistence.inMemory.enabled }} + emptyDir: + medium: Memory + {{- with .Values.persistence.inMemory.sizeLimit }} + sizeLimit: {{ . }} + {{- end }} + {{- else }} + emptyDir: {} + {{- end }} + {{- end }} + {{- if .Values.sidecar.alerts.enabled }} + - name: sc-alerts-volume + emptyDir: + {{- with .Values.sidecar.alerts.sizeLimit }} + sizeLimit: {{ . }} + {{- else }} + {} + {{- end }} + {{- end }} + {{- if .Values.sidecar.dashboards.enabled }} + - name: sc-dashboard-volume + emptyDir: + {{- with .Values.sidecar.dashboards.sizeLimit }} + sizeLimit: {{ . }} + {{- else }} + {} + {{- end }} + {{- if .Values.sidecar.dashboards.SCProvider }} + - name: sc-dashboard-provider + configMap: + name: {{ include "grafana.fullname" . }}-config-dashboards + {{- end }} + {{- end }} + {{- if .Values.sidecar.datasources.enabled }} + - name: sc-datasources-volume + emptyDir: + {{- with .Values.sidecar.datasources.sizeLimit }} + sizeLimit: {{ . }} + {{- else }} + {} + {{- end }} + {{- end }} + {{- if .Values.sidecar.plugins.enabled }} + - name: sc-plugins-volume + emptyDir: + {{- with .Values.sidecar.plugins.sizeLimit }} + sizeLimit: {{ . }} + {{- else }} + {} + {{- end }} + {{- end }} + {{- if .Values.sidecar.notifiers.enabled }} + - name: sc-notifiers-volume + emptyDir: + {{- with .Values.sidecar.notifiers.sizeLimit }} + sizeLimit: {{ . }} + {{- else }} + {} + {{- end }} + {{- end }} + {{- range .Values.extraSecretMounts }} + {{- if .secretName }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + defaultMode: {{ .defaultMode }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- with .items }} + items: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- else if .projected }} + - name: {{ .name }} + projected: + {{- toYaml .projected | nindent 6 }} + {{- else if .csi }} + - name: {{ .name }} + csi: + {{- toYaml .csi | nindent 6 }} + {{- end }} + {{- end }} + {{- range .Values.extraVolumes }} + - name: {{ .name }} + {{- if .existingClaim }} + persistentVolumeClaim: + claimName: {{ .existingClaim }} + {{- else if .hostPath }} + hostPath: + {{ toYaml .hostPath | nindent 6 }} + {{- else if .csi }} + csi: + {{- toYaml .csi | nindent 6 }} + {{- else if .configMap }} + configMap: + {{- toYaml .configMap | nindent 6 }} + {{- else if .emptyDir }} + emptyDir: + {{- toYaml .emptyDir | nindent 6 }} + {{- else }} + emptyDir: {} + {{- end }} + {{- end }} + {{- range .Values.extraEmptyDirMounts }} + - name: {{ .name }} + emptyDir: {} + {{- end }} + {{- with .Values.extraContainerVolumes }} + {{- tpl (toYaml .) $root | nindent 2 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrole.yaml new file mode 100644 index 000000000..3af4b62b6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrole.yaml @@ -0,0 +1,25 @@ +{{- if and .Values.rbac.create (or (not .Values.rbac.namespaced) .Values.rbac.extraClusterRoleRules) (not .Values.rbac.useExistingClusterRole) }} +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ include "grafana.fullname" . }}-clusterrole +{{- if or .Values.sidecar.dashboards.enabled .Values.rbac.extraClusterRoleRules .Values.sidecar.datasources.enabled .Values.sidecar.plugins.enabled .Values.sidecar.alerts.enabled }} +rules: + {{- if or .Values.sidecar.dashboards.enabled .Values.sidecar.datasources.enabled .Values.sidecar.plugins.enabled .Values.sidecar.alerts.enabled }} + - apiGroups: [""] # "" indicates the core API group + resources: ["configmaps", "secrets"] + verbs: ["get", "watch", "list"] + {{- end}} + {{- with .Values.rbac.extraClusterRoleRules }} + {{- toYaml . | nindent 2 }} + {{- end}} +{{- else }} +rules: [] +{{- end}} +{{- end}} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..bda9431a2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/clusterrolebinding.yaml @@ -0,0 +1,24 @@ +{{- if and .Values.rbac.create (or (not .Values.rbac.namespaced) .Values.rbac.extraClusterRoleRules) }} +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "grafana.fullname" . }}-clusterrolebinding + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +subjects: + - kind: ServiceAccount + name: {{ include "grafana.serviceAccountName" . }} + namespace: {{ include "grafana.namespace" . }} +roleRef: + kind: ClusterRole + {{- if .Values.rbac.useExistingClusterRole }} + name: {{ .Values.rbac.useExistingClusterRole }} + {{- else }} + name: {{ include "grafana.fullname" . }}-clusterrole + {{- end }} + apiGroup: rbac.authorization.k8s.io +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/configSecret.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configSecret.yaml new file mode 100644 index 000000000..55574b9bb --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configSecret.yaml @@ -0,0 +1,43 @@ +{{- $createConfigSecret := eq (include "grafana.shouldCreateConfigSecret" .) "true" -}} +{{- if and .Values.createConfigmap $createConfigSecret }} +{{- $files := .Files }} +{{- $root := . -}} +apiVersion: v1 +kind: Secret +metadata: + name: "{{ include "grafana.fullname" . }}-config-secret" + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: +{{- range $key, $value := .Values.alerting }} + {{- if (hasKey $value "secretFile") }} + {{- $key | nindent 2 }}: + {{- toYaml ( $files.Get $value.secretFile ) | b64enc | nindent 4}} + {{/* as of https://helm.sh/docs/chart_template_guide/accessing_files/ this will only work if you fork this chart and add files to it*/}} + {{- end }} +{{- end }} +stringData: +{{- range $key, $value := .Values.datasources }} +{{- if (hasKey $value "secret") }} +{{- $key | nindent 2 }}: | + {{- tpl (toYaml $value.secret | nindent 4) $root }} +{{- end }} +{{- end }} +{{- range $key, $value := .Values.notifiers }} +{{- if (hasKey $value "secret") }} +{{- $key | nindent 2 }}: | + {{- tpl (toYaml $value.secret | nindent 4) $root }} +{{- end }} +{{- end }} +{{- range $key, $value := .Values.alerting }} +{{ if (hasKey $value "secret") }} + {{- $key | nindent 2 }}: | + {{- tpl (toYaml $value.secret | nindent 4) $root }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap-dashboard-provider.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap-dashboard-provider.yaml new file mode 100644 index 000000000..b412c4d1f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap-dashboard-provider.yaml @@ -0,0 +1,15 @@ +{{- if and .Values.sidecar.dashboards.enabled .Values.sidecar.dashboards.SCProvider }} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ include "grafana.fullname" . }}-config-dashboards + namespace: {{ include "grafana.namespace" . }} +data: + {{- include "grafana.configDashboardProviderData" . | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap.yaml new file mode 100644 index 000000000..0a2edf47e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/configmap.yaml @@ -0,0 +1,20 @@ +{{- if .Values.createConfigmap }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- if or .Values.configMapAnnotations .Values.annotations }} + annotations: + {{- with .Values.annotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.configMapAnnotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- end }} +data: + {{- include "grafana.configData" . | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/dashboards-json-configmap.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/dashboards-json-configmap.yaml new file mode 100644 index 000000000..df0ed0d8c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/dashboards-json-configmap.yaml @@ -0,0 +1,35 @@ +{{- if .Values.dashboards }} +{{ $files := .Files }} +{{- range $provider, $dashboards := .Values.dashboards }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "grafana.fullname" $ }}-dashboards-{{ $provider }} + namespace: {{ include "grafana.namespace" $ }} + labels: + {{- include "grafana.labels" $ | nindent 4 }} + dashboard-provider: {{ $provider }} +{{- if $dashboards }} +data: +{{- $dashboardFound := false }} +{{- range $key, $value := $dashboards }} +{{- if (or (hasKey $value "json") (hasKey $value "file")) }} +{{- $dashboardFound = true }} + {{- print $key | nindent 2 }}.json: + {{- if hasKey $value "json" }} + |- + {{- $value.json | nindent 6 }} + {{- end }} + {{- if hasKey $value "file" }} + {{- toYaml ( $files.Get $value.file ) | nindent 4}} + {{- end }} +{{- end }} +{{- end }} +{{- if not $dashboardFound }} + {} +{{- end }} +{{- end }} +--- +{{- end }} + +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/deployment.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/deployment.yaml new file mode 100644 index 000000000..ad0d5680f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/deployment.yaml @@ -0,0 +1,54 @@ +{{- if (and (not .Values.useStatefulSet) (or (not .Values.persistence.enabled) (eq .Values.persistence.type "pvc"))) }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- if (not .Values.autoscaling.enabled) }} + replicas: {{ .Values.replicas }} + {{- end }} + revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + {{- with .Values.deploymentStrategy }} + strategy: + {{- toYaml . | trim | nindent 4 }} + {{- end }} + template: + metadata: + labels: + {{- include "grafana.selectorLabels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + annotations: + checksum/config: {{ include "grafana.configData" . | sha256sum }} + {{- if .Values.dashboards }} + checksum/dashboards-json-config: {{ include (print $.Template.BasePath "/dashboards-json-configmap.yaml") . | sha256sum }} + {{- end }} + checksum/sc-dashboard-provider-config: {{ include "grafana.configDashboardProviderData" . | sha256sum }} + {{- if and (or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret))) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + checksum/secret: {{ include "grafana.secretsData" . | sha256sum }} + {{- end }} + {{- if .Values.envRenderSecret }} + checksum/secret-env: {{ tpl (toYaml .Values.envRenderSecret) . | sha256sum }} + {{- end }} + kubectl.kubernetes.io/default-container: {{ .Chart.Name }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- include "grafana.pod" . | nindent 6 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/extra-manifests.yaml new file mode 100644 index 000000000..a9bb3b6ba --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraObjects }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/headless-service.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/headless-service.yaml new file mode 100644 index 000000000..3028589d3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/headless-service.yaml @@ -0,0 +1,22 @@ +{{- $sts := list "sts" "StatefulSet" "statefulset" -}} +{{- if or .Values.headlessService (and .Values.persistence.enabled (not .Values.persistence.existingClaim) (has .Values.persistence.type $sts)) }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "grafana.fullname" . }}-headless + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + clusterIP: None + selector: + {{- include "grafana.selectorLabels" . | nindent 4 }} + type: ClusterIP + ports: + - name: {{ .Values.gossipPortName }}-tcp + port: 9094 +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/hpa.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/hpa.yaml new file mode 100644 index 000000000..46bbcb49a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/hpa.yaml @@ -0,0 +1,52 @@ +{{- $sts := list "sts" "StatefulSet" "statefulset" -}} +{{- if .Values.autoscaling.enabled }} +apiVersion: {{ include "grafana.hpa.apiVersion" . }} +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + app.kubernetes.io/name: {{ include "grafana.name" . }} + helm.sh/chart: {{ include "grafana.chart" . }} + app.kubernetes.io/managed-by: {{ .Release.Service }} + app.kubernetes.io/instance: {{ .Release.Name }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + {{- if has .Values.persistence.type $sts }} + kind: StatefulSet + {{- else }} + kind: Deployment + {{- end }} + name: {{ include "grafana.fullname" . }} + minReplicas: {{ .Values.autoscaling.minReplicas }} + maxReplicas: {{ .Values.autoscaling.maxReplicas }} + metrics: + {{- if .Values.autoscaling.targetMemory }} + - type: Resource + resource: + name: memory + {{- if eq (include "grafana.hpa.apiVersion" .) "autoscaling/v2beta1" }} + targetAverageUtilization: {{ .Values.autoscaling.targetMemory }} + {{- else }} + target: + type: Utilization + averageUtilization: {{ .Values.autoscaling.targetMemory }} + {{- end }} + {{- end }} + {{- if .Values.autoscaling.targetCPU }} + - type: Resource + resource: + name: cpu + {{- if eq (include "grafana.hpa.apiVersion" .) "autoscaling/v2beta1" }} + targetAverageUtilization: {{ .Values.autoscaling.targetCPU }} + {{- else }} + target: + type: Utilization + averageUtilization: {{ .Values.autoscaling.targetCPU }} + {{- end }} + {{- end }} + {{- if .Values.autoscaling.behavior }} + behavior: {{ toYaml .Values.autoscaling.behavior | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-deployment.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-deployment.yaml new file mode 100644 index 000000000..7722ede50 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-deployment.yaml @@ -0,0 +1,200 @@ +{{ if .Values.imageRenderer.enabled }} +{{- $root := . -}} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.imageRenderer.labels" . | nindent 4 }} + {{- with .Values.imageRenderer.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.imageRenderer.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- if and (not .Values.imageRenderer.autoscaling.enabled) (.Values.imageRenderer.replicas) }} + replicas: {{ .Values.imageRenderer.replicas }} + {{- end }} + revisionHistoryLimit: {{ .Values.imageRenderer.revisionHistoryLimit }} + selector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + + {{- with .Values.imageRenderer.deploymentStrategy }} + strategy: + {{- toYaml . | trim | nindent 4 }} + {{- end }} + template: + metadata: + labels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 8 }} + {{- with .Values.imageRenderer.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + {{- with .Values.imageRenderer.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.imageRenderer.schedulerName }} + schedulerName: "{{ . }}" + {{- end }} + {{- with .Values.imageRenderer.serviceAccountName }} + serviceAccountName: "{{ . }}" + {{- end }} + {{- with .Values.imageRenderer.securityContext }} + securityContext: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.imageRenderer.hostAliases }} + hostAliases: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.imageRenderer.priorityClassName }} + priorityClassName: {{ . }} + {{- end }} + {{- with .Values.imageRenderer.image.pullSecrets }} + imagePullSecrets: + {{- range . }} + - name: {{ tpl . $root }} + {{- end}} + {{- end }} + containers: + - name: {{ .Chart.Name }}-image-renderer + {{- $registry := .Values.global.imageRegistry | default .Values.imageRenderer.image.registry -}} + {{- if .Values.imageRenderer.image.sha }} + image: "{{ $registry }}/{{ .Values.imageRenderer.image.repository }}:{{ .Values.imageRenderer.image.tag }}@sha256:{{ .Values.imageRenderer.image.sha }}" + {{- else }} + image: "{{ $registry }}/{{ .Values.imageRenderer.image.repository }}:{{ .Values.imageRenderer.image.tag }}" + {{- end }} + imagePullPolicy: {{ .Values.imageRenderer.image.pullPolicy }} + {{- if .Values.imageRenderer.command }} + command: + {{- range .Values.imageRenderer.command }} + - {{ . }} + {{- end }} + {{- end}} + ports: + - name: {{ .Values.imageRenderer.service.portName }} + containerPort: {{ .Values.imageRenderer.service.targetPort }} + protocol: TCP + livenessProbe: + httpGet: + path: / + port: {{ .Values.imageRenderer.service.portName }} + env: + - name: HTTP_PORT + value: {{ .Values.imageRenderer.service.targetPort | quote }} + {{- if .Values.imageRenderer.serviceMonitor.enabled }} + - name: ENABLE_METRICS + value: "true" + {{- end }} + {{- range $key, $value := .Values.imageRenderer.envValueFrom }} + - name: {{ $key | quote }} + valueFrom: + {{- tpl (toYaml $value) $ | nindent 16 }} + {{- end }} + {{- range $key, $value := .Values.imageRenderer.env }} + - name: {{ $key | quote }} + value: {{ $value | quote }} + {{- end }} + {{- with .Values.imageRenderer.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - mountPath: /tmp + name: image-renderer-tmpfs + {{- range .Values.imageRenderer.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + mountPath: {{ tpl .mountPath $root }} + subPath: {{ tpl (.subPath | default "") $root }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.imageRenderer.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + subPath: {{ .subPath | default "" }} + {{- end }} + {{- range .Values.imageRenderer.extraVolumeMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath | default "" }} + readOnly: {{ .readOnly }} + {{- end }} + {{- with .Values.imageRenderer.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.imageRenderer.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.imageRenderer.affinity }} + affinity: + {{- tpl (toYaml .) $root | nindent 8 }} + {{- end }} + {{- with .Values.imageRenderer.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: image-renderer-tmpfs + emptyDir: {} + {{- range .Values.imageRenderer.extraConfigmapMounts }} + - name: {{ tpl .name $root }} + configMap: + name: {{ tpl .configMap $root }} + {{- with .items }} + items: + {{- toYaml . | nindent 14 }} + {{- end }} + {{- end }} + {{- range .Values.imageRenderer.extraSecretMounts }} + {{- if .secretName }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + defaultMode: {{ .defaultMode }} + {{- with .items }} + items: + {{- toYaml . | nindent 14 }} + {{- end }} + {{- else if .projected }} + - name: {{ .name }} + projected: + {{- toYaml .projected | nindent 12 }} + {{- else if .csi }} + - name: {{ .name }} + csi: + {{- toYaml .csi | nindent 12 }} + {{- end }} + {{- end }} + {{- range .Values.imageRenderer.extraVolumes }} + - name: {{ .name }} + {{- if .existingClaim }} + persistentVolumeClaim: + claimName: {{ .existingClaim }} + {{- else if .hostPath }} + hostPath: + {{ toYaml .hostPath | nindent 12 }} + {{- else if .csi }} + csi: + {{- toYaml .csi | nindent 12 }} + {{- else if .configMap }} + configMap: + {{- toYaml .configMap | nindent 12 }} + {{- else if .emptyDir }} + emptyDir: + {{- toYaml .emptyDir | nindent 12 }} + {{- else }} + emptyDir: {} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-hpa.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-hpa.yaml new file mode 100644 index 000000000..b0f0059b7 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-hpa.yaml @@ -0,0 +1,47 @@ +{{- if and .Values.imageRenderer.enabled .Values.imageRenderer.autoscaling.enabled }} +apiVersion: {{ include "grafana.hpa.apiVersion" . }} +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer + namespace: {{ include "grafana.namespace" . }} + labels: + app.kubernetes.io/name: {{ include "grafana.name" . }}-image-renderer + helm.sh/chart: {{ include "grafana.chart" . }} + app.kubernetes.io/managed-by: {{ .Release.Service }} + app.kubernetes.io/instance: {{ .Release.Name }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "grafana.fullname" . }}-image-renderer + minReplicas: {{ .Values.imageRenderer.autoscaling.minReplicas }} + maxReplicas: {{ .Values.imageRenderer.autoscaling.maxReplicas }} + metrics: + {{- if .Values.imageRenderer.autoscaling.targetMemory }} + - type: Resource + resource: + name: memory + {{- if eq (include "grafana.hpa.apiVersion" .) "autoscaling/v2beta1" }} + targetAverageUtilization: {{ .Values.imageRenderer.autoscaling.targetMemory }} + {{- else }} + target: + type: Utilization + averageUtilization: {{ .Values.imageRenderer.autoscaling.targetMemory }} + {{- end }} + {{- end }} + {{- if .Values.imageRenderer.autoscaling.targetCPU }} + - type: Resource + resource: + name: cpu + {{- if eq (include "grafana.hpa.apiVersion" .) "autoscaling/v2beta1" }} + targetAverageUtilization: {{ .Values.imageRenderer.autoscaling.targetCPU }} + {{- else }} + target: + type: Utilization + averageUtilization: {{ .Values.imageRenderer.autoscaling.targetCPU }} + {{- end }} + {{- end }} + {{- if .Values.imageRenderer.autoscaling.behavior }} + behavior: {{ toYaml .Values.imageRenderer.autoscaling.behavior | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-network-policy.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-network-policy.yaml new file mode 100644 index 000000000..bcbd24976 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-network-policy.yaml @@ -0,0 +1,79 @@ +{{- if and .Values.imageRenderer.enabled .Values.imageRenderer.networkPolicy.limitIngress }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer-ingress + namespace: {{ include "grafana.namespace" . }} + annotations: + comment: Limit image-renderer ingress traffic from grafana +spec: + podSelector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + {{- with .Values.imageRenderer.podLabels }} + {{- toYaml . | nindent 6 }} + {{- end }} + + policyTypes: + - Ingress + ingress: + - ports: + - port: {{ .Values.imageRenderer.service.targetPort }} + protocol: TCP + from: + - namespaceSelector: + matchLabels: + kubernetes.io/metadata.name: {{ include "grafana.namespace" . }} + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 14 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 14 }} + {{- end }} + {{- with .Values.imageRenderer.networkPolicy.extraIngressSelectors -}} + {{ toYaml . | nindent 8 }} + {{- end }} +{{- end }} + +{{- if and .Values.imageRenderer.enabled .Values.imageRenderer.networkPolicy.limitEgress }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer-egress + namespace: {{ include "grafana.namespace" . }} + annotations: + comment: Limit image-renderer egress traffic to grafana +spec: + podSelector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + {{- with .Values.imageRenderer.podLabels }} + {{- toYaml . | nindent 6 }} + {{- end }} + + policyTypes: + - Egress + egress: + # allow dns resolution + - ports: + - port: 53 + protocol: UDP + - port: 53 + protocol: TCP + # talk only to grafana + - ports: + - port: {{ .Values.service.targetPort }} + protocol: TCP + to: + - namespaceSelector: + matchLabels: + kubernetes.io/metadata.name: {{ include "grafana.namespace" . }} + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 14 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 14 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-service.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-service.yaml new file mode 100644 index 000000000..f8da127cf --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-service.yaml @@ -0,0 +1,31 @@ +{{- if and .Values.imageRenderer.enabled .Values.imageRenderer.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.imageRenderer.labels" . | nindent 4 }} + {{- with .Values.imageRenderer.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.imageRenderer.service.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + type: ClusterIP + {{- with .Values.imageRenderer.service.clusterIP }} + clusterIP: {{ . }} + {{- end }} + ports: + - name: {{ .Values.imageRenderer.service.portName }} + port: {{ .Values.imageRenderer.service.port }} + protocol: TCP + targetPort: {{ .Values.imageRenderer.service.targetPort }} + {{- with .Values.imageRenderer.appProtocol }} + appProtocol: {{ . }} + {{- end }} + selector: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-servicemonitor.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-servicemonitor.yaml new file mode 100644 index 000000000..5d9f09d26 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/image-renderer-servicemonitor.yaml @@ -0,0 +1,48 @@ +{{- if .Values.imageRenderer.serviceMonitor.enabled }} +--- +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "grafana.fullname" . }}-image-renderer + {{- if .Values.imageRenderer.serviceMonitor.namespace }} + namespace: {{ tpl .Values.imageRenderer.serviceMonitor.namespace . }} + {{- else }} + namespace: {{ include "grafana.namespace" . }} + {{- end }} + labels: + {{- include "grafana.imageRenderer.labels" . | nindent 4 }} + {{- with .Values.imageRenderer.serviceMonitor.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + endpoints: + - port: {{ .Values.imageRenderer.service.portName }} + {{- with .Values.imageRenderer.serviceMonitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.imageRenderer.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + honorLabels: true + path: {{ .Values.imageRenderer.serviceMonitor.path }} + scheme: {{ .Values.imageRenderer.serviceMonitor.scheme }} + {{- with .Values.imageRenderer.serviceMonitor.tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.imageRenderer.serviceMonitor.relabelings }} + relabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + jobLabel: "{{ .Release.Name }}-image-renderer" + selector: + matchLabels: + {{- include "grafana.imageRenderer.selectorLabels" . | nindent 6 }} + namespaceSelector: + matchNames: + - {{ include "grafana.namespace" . }} + {{- with .Values.imageRenderer.serviceMonitor.targetLabels }} + targetLabels: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/ingress.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/ingress.yaml new file mode 100644 index 000000000..b2ffd8109 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/ingress.yaml @@ -0,0 +1,78 @@ +{{- if .Values.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "grafana.ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "grafana.ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "grafana.ingress.supportsPathType" .) "true" -}} +{{- $fullName := include "grafana.fullname" . -}} +{{- $servicePort := .Values.service.port -}} +{{- $ingressPath := .Values.ingress.path -}} +{{- $ingressPathType := .Values.ingress.pathType -}} +{{- $extraPaths := .Values.ingress.extraPaths -}} +apiVersion: {{ include "grafana.ingress.apiVersion" . }} +kind: Ingress +metadata: + name: {{ $fullName }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.ingress.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.ingress.annotations }} + annotations: + {{- range $key, $value := . }} + {{ $key }}: {{ tpl $value $ | quote }} + {{- end }} + {{- end }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.ingress.ingressClassName }} + ingressClassName: {{ .Values.ingress.ingressClassName }} + {{- end -}} + {{- with .Values.ingress.tls }} + tls: + {{- tpl (toYaml .) $ | nindent 4 }} + {{- end }} + rules: + {{- if .Values.ingress.hosts }} + {{- range .Values.ingress.hosts }} + - host: {{ tpl . $ | quote }} + http: + paths: + {{- with $extraPaths }} + {{- toYaml . | nindent 10 }} + {{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $fullName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $fullName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end }} + {{- else }} + - http: + paths: + - backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $fullName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $fullName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- with $ingressPath }} + path: {{ . }} + {{- end }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + {{- end -}} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/networkpolicy.yaml new file mode 100644 index 000000000..4cd3ed697 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/networkpolicy.yaml @@ -0,0 +1,61 @@ +{{- if .Values.networkPolicy.enabled }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + policyTypes: + {{- if .Values.networkPolicy.ingress }} + - Ingress + {{- end }} + {{- if .Values.networkPolicy.egress.enabled }} + - Egress + {{- end }} + podSelector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + + {{- if .Values.networkPolicy.egress.enabled }} + egress: + {{- if not .Values.networkPolicy.egress.blockDNSResolution }} + - ports: + - port: 53 + protocol: UDP + {{- end }} + - ports: + {{ .Values.networkPolicy.egress.ports | toJson }} + {{- with .Values.networkPolicy.egress.to }} + to: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- end }} + {{- if .Values.networkPolicy.ingress }} + ingress: + - ports: + - port: {{ .Values.service.targetPort }} + {{- if not .Values.networkPolicy.allowExternal }} + from: + - podSelector: + matchLabels: + {{ include "grafana.fullname" . }}-client: "true" + {{- with .Values.networkPolicy.explicitNamespacesSelector }} + - namespaceSelector: + {{- toYaml . | nindent 12 }} + {{- end }} + - podSelector: + matchLabels: + {{- include "grafana.labels" . | nindent 14 }} + role: read + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/poddisruptionbudget.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/poddisruptionbudget.yaml new file mode 100644 index 000000000..05251214a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/poddisruptionbudget.yaml @@ -0,0 +1,22 @@ +{{- if .Values.podDisruptionBudget }} +apiVersion: {{ include "grafana.podDisruptionBudget.apiVersion" . }} +kind: PodDisruptionBudget +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- with .Values.podDisruptionBudget.minAvailable }} + minAvailable: {{ . }} + {{- end }} + {{- with .Values.podDisruptionBudget.maxUnavailable }} + maxUnavailable: {{ . }} + {{- end }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..eed7af95b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/podsecuritypolicy.yaml @@ -0,0 +1,49 @@ +{{- if and .Values.rbac.pspEnabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "grafana.fullname" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + annotations: + seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default,runtime/default' + seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + {{- if .Values.rbac.pspUseAppArmor }} + apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default' + apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + {{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + # Default set from Docker, with DAC_OVERRIDE and CHOWN + - ALL + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'csi' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/pvc.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/pvc.yaml new file mode 100644 index 000000000..d1c4b2de2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/pvc.yaml @@ -0,0 +1,39 @@ +{{- if and (not .Values.useStatefulSet) .Values.persistence.enabled (not .Values.persistence.existingClaim) (eq .Values.persistence.type "pvc")}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.persistence.extraPvcLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.persistence.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.persistence.finalizers }} + finalizers: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{- if and (.Values.persistence.lookupVolumeName) (lookup "v1" "PersistentVolumeClaim" (include "grafana.namespace" .) (include "grafana.fullname" .)) }} + volumeName: {{ (lookup "v1" "PersistentVolumeClaim" (include "grafana.namespace" .) (include "grafana.fullname" .)).spec.volumeName }} + {{- end }} + {{- with .Values.persistence.storageClassName }} + storageClassName: {{ . }} + {{- end }} + {{- with .Values.persistence.selectorLabels }} + selector: + matchLabels: + {{- toYaml . | nindent 6 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/role.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/role.yaml new file mode 100644 index 000000000..4b5edd978 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/role.yaml @@ -0,0 +1,32 @@ +{{- if and .Values.rbac.create (not .Values.rbac.useExistingRole) -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- if or .Values.rbac.pspEnabled (and .Values.rbac.namespaced (or .Values.sidecar.dashboards.enabled .Values.sidecar.datasources.enabled .Values.sidecar.plugins.enabled .Values.rbac.extraRoleRules)) }} +rules: + {{- if and .Values.rbac.pspEnabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} + - apiGroups: ['extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: [{{ include "grafana.fullname" . }}] + {{- end }} + {{- if and .Values.rbac.namespaced (or .Values.sidecar.dashboards.enabled .Values.sidecar.datasources.enabled .Values.sidecar.plugins.enabled) }} + - apiGroups: [""] # "" indicates the core API group + resources: ["configmaps", "secrets"] + verbs: ["get", "watch", "list"] + {{- end }} + {{- with .Values.rbac.extraRoleRules }} + {{- toYaml . | nindent 2 }} + {{- end}} +{{- else }} +rules: [] +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/rolebinding.yaml new file mode 100644 index 000000000..58f77c6b0 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/rolebinding.yaml @@ -0,0 +1,25 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + {{- if .Values.rbac.useExistingRole }} + name: {{ .Values.rbac.useExistingRole }} + {{- else }} + name: {{ include "grafana.fullname" . }} + {{- end }} +subjects: +- kind: ServiceAccount + name: {{ include "grafana.serviceAccountName" . }} + namespace: {{ include "grafana.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret-env.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret-env.yaml new file mode 100644 index 000000000..eb14aac70 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret-env.yaml @@ -0,0 +1,14 @@ +{{- if .Values.envRenderSecret }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "grafana.fullname" . }}-env + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} +type: Opaque +data: +{{- range $key, $val := .Values.envRenderSecret }} + {{ $key }}: {{ tpl ($val | toString) $ | b64enc | quote }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret.yaml new file mode 100644 index 000000000..fd2ca50f4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/secret.yaml @@ -0,0 +1,16 @@ +{{- if or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret)) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +type: Opaque +data: + {{- include "grafana.secretsData" . | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/service.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/service.yaml new file mode 100644 index 000000000..022328c11 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/service.yaml @@ -0,0 +1,67 @@ +{{- if .Values.service.enabled }} +{{- $root := . }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.service.annotations }} + annotations: + {{- tpl (toYaml . | nindent 4) $root }} + {{- end }} +spec: + {{- if (or (eq .Values.service.type "ClusterIP") (empty .Values.service.type)) }} + type: ClusterIP + {{- with .Values.service.clusterIP }} + clusterIP: {{ . }} + {{- end }} + {{- else if eq .Values.service.type "LoadBalancer" }} + type: LoadBalancer + {{- with .Values.service.loadBalancerIP }} + loadBalancerIP: {{ . }} + {{- end }} + {{- with .Values.service.loadBalancerClass }} + loadBalancerClass: {{ . }} + {{- end }} + {{- with .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- else }} + type: {{ .Values.service.type }} + {{- end }} + {{- if .Values.service.ipFamilyPolicy }} + ipFamilyPolicy: {{ .Values.service.ipFamilyPolicy }} + {{- end }} + {{- if .Values.service.ipFamilies }} + ipFamilies: {{ .Values.service.ipFamilies | toYaml | nindent 2 }} + {{- end }} + {{- with .Values.service.externalIPs }} + externalIPs: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.service.externalTrafficPolicy }} + externalTrafficPolicy: {{ . }} + {{- end }} + ports: + - name: {{ .Values.service.portName }} + port: {{ .Values.service.port }} + protocol: TCP + targetPort: {{ .Values.service.targetPort }} + {{- with .Values.service.appProtocol }} + appProtocol: {{ . }} + {{- end }} + {{- if (and (eq .Values.service.type "NodePort") (not (empty .Values.service.nodePort))) }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + {{- with .Values.extraExposePorts }} + {{- tpl (toYaml . | nindent 4) $root }} + {{- end }} + selector: + {{- include "grafana.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/serviceaccount.yaml new file mode 100644 index 000000000..ffca0717a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/serviceaccount.yaml @@ -0,0 +1,17 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +automountServiceAccountToken: {{ .Values.serviceAccount.autoMount | default .Values.serviceAccount.automountServiceAccountToken }} +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.serviceAccount.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- tpl (toYaml . | nindent 4) $ }} + {{- end }} + name: {{ include "grafana.serviceAccountName" . }} + namespace: {{ include "grafana.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/servicemonitor.yaml new file mode 100644 index 000000000..035901352 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/servicemonitor.yaml @@ -0,0 +1,52 @@ +{{- if .Values.serviceMonitor.enabled }} +--- +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "grafana.fullname" . }} + {{- if .Values.serviceMonitor.namespace }} + namespace: {{ tpl .Values.serviceMonitor.namespace . }} + {{- else }} + namespace: {{ include "grafana.namespace" . }} + {{- end }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.serviceMonitor.labels }} + {{- tpl (toYaml . | nindent 4) $ }} + {{- end }} +spec: + endpoints: + - port: {{ .Values.service.portName }} + {{- with .Values.serviceMonitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + honorLabels: true + path: {{ .Values.serviceMonitor.path }} + scheme: {{ .Values.serviceMonitor.scheme }} + {{- with .Values.serviceMonitor.tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.serviceMonitor.relabelings }} + relabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.serviceMonitor.metricRelabelings }} + metricRelabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + jobLabel: "{{ .Release.Name }}" + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + namespaceSelector: + matchNames: + - {{ include "grafana.namespace" . }} + {{- with .Values.serviceMonitor.targetLabels }} + targetLabels: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/statefulset.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/statefulset.yaml new file mode 100644 index 000000000..7546c1887 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/statefulset.yaml @@ -0,0 +1,58 @@ +{{- $sts := list "sts" "StatefulSet" "statefulset" -}} +{{- if (or (.Values.useStatefulSet) (and .Values.persistence.enabled (not .Values.persistence.existingClaim) (has .Values.persistence.type $sts)))}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: + name: {{ include "grafana.fullname" . }} + namespace: {{ include "grafana.namespace" . }} + labels: + {{- include "grafana.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + replicas: {{ .Values.replicas }} + selector: + matchLabels: + {{- include "grafana.selectorLabels" . | nindent 6 }} + serviceName: {{ include "grafana.fullname" . }}-headless + template: + metadata: + labels: + {{- include "grafana.selectorLabels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + checksum/dashboards-json-config: {{ include (print $.Template.BasePath "/dashboards-json-configmap.yaml") . | sha256sum }} + checksum/sc-dashboard-provider-config: {{ include (print $.Template.BasePath "/configmap-dashboard-provider.yaml") . | sha256sum }} + {{- if and (or (and (not .Values.admin.existingSecret) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD__FILE) (not .Values.env.GF_SECURITY_ADMIN_PASSWORD)) (and .Values.ldap.enabled (not .Values.ldap.existingSecret))) (not .Values.env.GF_SECURITY_DISABLE_INITIAL_ADMIN_CREATION) }} + checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- end }} + kubectl.kubernetes.io/default-container: {{ .Chart.Name }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- include "grafana.pod" . | nindent 6 }} + {{- if .Values.persistence.enabled}} + volumeClaimTemplates: + - apiVersion: v1 + kind: PersistentVolumeClaim + metadata: + name: storage + spec: + accessModes: {{ .Values.persistence.accessModes }} + storageClassName: {{ .Values.persistence.storageClassName }} + resources: + requests: + storage: {{ .Values.persistence.size }} + {{- with .Values.persistence.selectorLabels }} + selector: + matchLabels: + {{- toYaml . | nindent 10 }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-configmap.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-configmap.yaml new file mode 100644 index 000000000..1e81bee90 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-configmap.yaml @@ -0,0 +1,20 @@ +{{- if .Values.testFramework.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "grafana.fullname" . }}-test + namespace: {{ include "grafana.namespace" . }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + labels: + {{- include "grafana.labels" . | nindent 4 }} +data: + run.sh: |- + @test "Test Health" { + url="http://{{ include "grafana.fullname" . }}/api/health" + + code=$(wget --server-response --spider --timeout 90 --tries 10 ${url} 2>&1 | awk '/^ HTTP/{print $2}') + [ "$code" == "200" ] + } +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-podsecuritypolicy.yaml new file mode 100644 index 000000000..c13a3bf66 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-podsecuritypolicy.yaml @@ -0,0 +1,32 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.testFramework.enabled .Values.rbac.pspEnabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "grafana.fullname" . }}-test + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + labels: + {{- include "grafana.labels" . | nindent 4 }} +spec: + allowPrivilegeEscalation: true + privileged: false + hostNetwork: false + hostIPC: false + hostPID: false + fsGroup: + rule: RunAsAny + seLinux: + rule: RunAsAny + supplementalGroups: + rule: RunAsAny + runAsUser: + rule: RunAsAny + volumes: + - configMap + - downwardAPI + - emptyDir + - projected + - csi + - secret +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-role.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-role.yaml new file mode 100644 index 000000000..75dddfdd3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-role.yaml @@ -0,0 +1,17 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.testFramework.enabled .Values.rbac.pspEnabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "grafana.fullname" . }}-test + namespace: {{ include "grafana.namespace" . }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + labels: + {{- include "grafana.labels" . | nindent 4 }} +rules: + - apiGroups: ['policy'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: [{{ include "grafana.fullname" . }}-test] +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-rolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-rolebinding.yaml new file mode 100644 index 000000000..c0d2d39ef --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.testFramework.enabled .Values.rbac.pspEnabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "grafana.fullname" . }}-test + namespace: {{ include "grafana.namespace" . }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + labels: + {{- include "grafana.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "grafana.fullname" . }}-test +subjects: + - kind: ServiceAccount + name: {{ include "grafana.serviceAccountNameTest" . }} + namespace: {{ include "grafana.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-serviceaccount.yaml new file mode 100644 index 000000000..7af898272 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test-serviceaccount.yaml @@ -0,0 +1,12 @@ +{{- if and .Values.testFramework.enabled .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "grafana.labels" . | nindent 4 }} + name: {{ include "grafana.serviceAccountNameTest" . }} + namespace: {{ include "grafana.namespace" . }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test.yaml new file mode 100644 index 000000000..2484a96da --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/templates/tests/test.yaml @@ -0,0 +1,53 @@ +{{- if .Values.testFramework.enabled }} +{{- $root := . }} +apiVersion: v1 +kind: Pod +metadata: + name: {{ include "grafana.fullname" . }}-test + labels: + {{- include "grafana.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": test + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + namespace: {{ include "grafana.namespace" . }} +spec: + serviceAccountName: {{ include "grafana.serviceAccountNameTest" . }} + {{- with .Values.testFramework.securityContext }} + securityContext: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- if or .Values.image.pullSecrets .Values.global.imagePullSecrets }} + imagePullSecrets: + {{- include "grafana.imagePullSecrets" (dict "root" $root "imagePullSecrets" .Values.image.pullSecrets) | nindent 4 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.affinity }} + affinity: + {{- tpl (toYaml .) $root | nindent 4 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 4 }} + {{- end }} + containers: + - name: {{ .Release.Name }}-test + image: "{{ .Values.global.imageRegistry | default .Values.testFramework.image.registry }}/{{ .Values.testFramework.image.repository }}:{{ .Values.testFramework.image.tag }}" + imagePullPolicy: "{{ .Values.testFramework.imagePullPolicy}}" + command: ["/opt/bats/bin/bats", "-t", "/tests/run.sh"] + volumeMounts: + - mountPath: /tests + name: tests + readOnly: true + {{- with .Values.testFramework.resources }} + resources: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: tests + configMap: + name: {{ include "grafana.fullname" . }}-test + restartPolicy: Never +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/grafana/values.yaml b/charts/kasten/k10/7.0.1401/charts/grafana/values.yaml new file mode 100644 index 000000000..6beae1bf4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/grafana/values.yaml @@ -0,0 +1,1392 @@ +global: + # -- Overrides the Docker registry globally for all images + imageRegistry: null + + # To help compatibility with other charts which use global.imagePullSecrets. + # Allow either an array of {name: pullSecret} maps (k8s-style), or an array of strings (more common helm-style). + # Can be templated. + # global: + # imagePullSecrets: + # - name: pullSecret1 + # - name: pullSecret2 + # or + # global: + # imagePullSecrets: + # - pullSecret1 + # - pullSecret2 + imagePullSecrets: [] + +rbac: + create: true + ## Use an existing ClusterRole/Role (depending on rbac.namespaced false/true) + # useExistingRole: name-of-some-role + # useExistingClusterRole: name-of-some-clusterRole + pspEnabled: false + pspUseAppArmor: false + namespaced: false + extraRoleRules: [] + # - apiGroups: [] + # resources: [] + # verbs: [] + extraClusterRoleRules: [] + # - apiGroups: [] + # resources: [] + # verbs: [] +serviceAccount: + create: true + name: + nameTest: + ## ServiceAccount labels. + labels: {} + ## Service account annotations. Can be templated. + # annotations: + # eks.amazonaws.com/role-arn: arn:aws:iam::123456789000:role/iam-role-name-here + + ## autoMount is deprecated in favor of automountServiceAccountToken + # autoMount: false + automountServiceAccountToken: false + +replicas: 1 + +## Create a headless service for the deployment +headlessService: false + +## Should the service account be auto mounted on the pod +automountServiceAccountToken: true + +## Create HorizontalPodAutoscaler object for deployment type +# +autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 5 + targetCPU: "60" + targetMemory: "" + behavior: {} + +## See `kubectl explain poddisruptionbudget.spec` for more +## ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ +podDisruptionBudget: {} +# apiVersion: "" +# minAvailable: 1 +# maxUnavailable: 1 + +## See `kubectl explain deployment.spec.strategy` for more +## ref: https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy +deploymentStrategy: + type: RollingUpdate + +readinessProbe: + httpGet: + path: /api/health + port: 3000 + +livenessProbe: + httpGet: + path: /api/health + port: 3000 + initialDelaySeconds: 60 + timeoutSeconds: 30 + failureThreshold: 10 + +## Use an alternate scheduler, e.g. "stork". +## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ +## +# schedulerName: "default-scheduler" + +image: + # -- The Docker registry + registry: docker.io + # -- Docker image repository + repository: grafana/grafana + # Overrides the Grafana image tag whose default is the chart appVersion + tag: "" + sha: "" + pullPolicy: IfNotPresent + + ## Optionally specify an array of imagePullSecrets. + ## Secrets must be manually created in the namespace. + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + ## Can be templated. + ## + pullSecrets: [] + # - myRegistrKeySecretName + +testFramework: + enabled: true + image: + # -- The Docker registry + registry: docker.io + repository: bats/bats + tag: "v1.4.1" + imagePullPolicy: IfNotPresent + securityContext: {} + resources: {} + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +# dns configuration for pod +dnsPolicy: ~ +dnsConfig: {} + # nameservers: + # - 8.8.8.8 + # options: + # - name: ndots + # value: "2" + # - name: edns0 + +securityContext: + runAsNonRoot: true + runAsUser: 472 + runAsGroup: 472 + fsGroup: 472 + +containerSecurityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + seccompProfile: + type: RuntimeDefault + +# Enable creating the grafana configmap +createConfigmap: true + +# Extra configmaps to mount in grafana pods +# Values are templated. +extraConfigmapMounts: [] + # - name: certs-configmap + # mountPath: /etc/grafana/ssl/ + # subPath: certificates.crt # (optional) + # configMap: certs-configmap + # readOnly: true + # optional: false + + +extraEmptyDirMounts: [] + # - name: provisioning-notifiers + # mountPath: /etc/grafana/provisioning/notifiers + + +# Apply extra labels to common labels. +extraLabels: {} + +## Assign a PriorityClassName to pods if set +# priorityClassName: + +downloadDashboardsImage: + # -- The Docker registry + registry: docker.io + repository: curlimages/curl + tag: 7.85.0 + sha: "" + pullPolicy: IfNotPresent + +downloadDashboards: + env: {} + envFromSecret: "" + resources: {} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + seccompProfile: + type: RuntimeDefault + envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + +## Pod Annotations +# podAnnotations: {} + +## ConfigMap Annotations +# configMapAnnotations: {} + # argocd.argoproj.io/sync-options: Replace=true + +## Pod Labels +# podLabels: {} + +podPortName: grafana +gossipPortName: gossip +## Deployment annotations +# annotations: {} + +## Expose the grafana service to be accessed from outside the cluster (LoadBalancer service). +## or access it from within the cluster (ClusterIP service). Set the service type and the port to serve it. +## ref: http://kubernetes.io/docs/user-guide/services/ +## +service: + enabled: true + type: ClusterIP + # Set the ip family policy to configure dual-stack see [Configure dual-stack](https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services) + ipFamilyPolicy: "" + # Sets the families that should be supported and the order in which they should be applied to ClusterIP as well. Can be IPv4 and/or IPv6. + ipFamilies: [] + loadBalancerIP: "" + loadBalancerClass: "" + loadBalancerSourceRanges: [] + port: 80 + targetPort: 3000 + # targetPort: 4181 To be used with a proxy extraContainer + ## Service annotations. Can be templated. + annotations: {} + labels: {} + portName: service + # Adds the appProtocol field to the service. This allows to work with istio protocol selection. Ex: "http" or "tcp" + appProtocol: "" + +serviceMonitor: + ## If true, a ServiceMonitor CR is created for a prometheus operator + ## https://github.com/coreos/prometheus-operator + ## + enabled: false + path: /metrics + # namespace: monitoring (defaults to use the namespace this chart is deployed to) + labels: {} + interval: 30s + scheme: http + tlsConfig: {} + scrapeTimeout: 30s + relabelings: [] + metricRelabelings: [] + targetLabels: [] + +extraExposePorts: [] + # - name: keycloak + # port: 8080 + # targetPort: 8080 + +# overrides pod.spec.hostAliases in the grafana deployment's pods +hostAliases: [] + # - ip: "1.2.3.4" + # hostnames: + # - "my.host.com" + +ingress: + enabled: false + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + # Values can be templated + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + labels: {} + path: / + + # pathType is only for k8s >= 1.1= + pathType: Prefix + + hosts: + - chart-example.local + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + ## Or for k8s > 1.19 + # - path: /* + # pathType: Prefix + # backend: + # service: + # name: ssl-redirect + # port: + # name: use-annotation + + + tls: [] + # - secretName: chart-example-tls + # hosts: + # - chart-example.local + +resources: {} +# limits: +# cpu: 100m +# memory: 128Mi +# requests: +# cpu: 100m +# memory: 128Mi + +## Node labels for pod assignment +## ref: https://kubernetes.io/docs/user-guide/node-selection/ +# +nodeSelector: {} + +## Tolerations for pod assignment +## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +## +tolerations: [] + +## Affinity for pod assignment (evaluated as template) +## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity +## +affinity: {} + +## Topology Spread Constraints +## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ +## +topologySpreadConstraints: [] + +## Additional init containers (evaluated as template) +## ref: https://kubernetes.io/docs/concepts/workloads/pods/init-containers/ +## +extraInitContainers: [] + +## Enable an Specify container in extraContainers. This is meant to allow adding an authentication proxy to a grafana pod +extraContainers: "" +# extraContainers: | +# - name: proxy +# image: quay.io/gambol99/keycloak-proxy:latest +# args: +# - -provider=github +# - -client-id= +# - -client-secret= +# - -github-org= +# - -email-domain=* +# - -cookie-secret= +# - -http-address=http://0.0.0.0:4181 +# - -upstream-url=http://127.0.0.1:3000 +# ports: +# - name: proxy-web +# containerPort: 4181 + +## Volumes that can be used in init containers that will not be mounted to deployment pods +extraContainerVolumes: [] +# - name: volume-from-secret +# secret: +# secretName: secret-to-mount +# - name: empty-dir-volume +# emptyDir: {} + +## Enable persistence using Persistent Volume Claims +## ref: https://kubernetes.io/docs/user-guide/persistent-volumes/ +## +persistence: + type: pvc + enabled: false + # storageClassName: default + accessModes: + - ReadWriteOnce + size: 10Gi + # annotations: {} + finalizers: + - kubernetes.io/pvc-protection + # selectorLabels: {} + ## Sub-directory of the PV to mount. Can be templated. + # subPath: "" + ## Name of an existing PVC. Can be templated. + # existingClaim: + ## Extra labels to apply to a PVC. + extraPvcLabels: {} + disableWarning: false + + ## If persistence is not enabled, this allows to mount the + ## local storage in-memory to improve performance + ## + inMemory: + enabled: false + ## The maximum usage on memory medium EmptyDir would be + ## the minimum value between the SizeLimit specified + ## here and the sum of memory limits of all containers in a pod + ## + # sizeLimit: 300Mi + + ## If 'lookupVolumeName' is set to true, Helm will attempt to retrieve + ## the current value of 'spec.volumeName' and incorporate it into the template. + lookupVolumeName: true + +initChownData: + ## If false, data ownership will not be reset at startup + ## This allows the grafana-server to be run with an arbitrary user + ## + enabled: true + + ## initChownData container image + ## + image: + # -- The Docker registry + registry: docker.io + repository: library/busybox + tag: "1.31.1" + sha: "" + pullPolicy: IfNotPresent + + ## initChownData resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: {} + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + securityContext: + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + capabilities: + add: + - CHOWN + +# Administrator credentials when not using an existing secret (see below) +adminUser: admin +# adminPassword: strongpassword + +# Use an existing secret for the admin user. +admin: + ## Name of the secret. Can be templated. + existingSecret: "" + userKey: admin-user + passwordKey: admin-password + +## Define command to be executed at startup by grafana container +## Needed if using `vault-env` to manage secrets (ref: https://banzaicloud.com/blog/inject-secrets-into-pods-vault/) +## Default is "run.sh" as defined in grafana's Dockerfile +# command: +# - "sh" +# - "/run.sh" + +## Optionally define args if command is used +## Needed if using `hashicorp/envconsul` to manage secrets +## By default no arguments are set +# args: +# - "-secret" +# - "secret/grafana" +# - "./grafana" + +## Extra environment variables that will be pass onto deployment pods +## +## to provide grafana with access to CloudWatch on AWS EKS: +## 1. create an iam role of type "Web identity" with provider oidc.eks.* (note the provider for later) +## 2. edit the "Trust relationships" of the role, add a line inside the StringEquals clause using the +## same oidc eks provider as noted before (same as the existing line) +## also, replace NAMESPACE and prometheus-operator-grafana with the service account namespace and name +## +## "oidc.eks.us-east-1.amazonaws.com/id/XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX:sub": "system:serviceaccount:NAMESPACE:prometheus-operator-grafana", +## +## 3. attach a policy to the role, you can use a built in policy called CloudWatchReadOnlyAccess +## 4. use the following env: (replace 123456789000 and iam-role-name-here with your aws account number and role name) +## +## env: +## AWS_ROLE_ARN: arn:aws:iam::123456789000:role/iam-role-name-here +## AWS_WEB_IDENTITY_TOKEN_FILE: /var/run/secrets/eks.amazonaws.com/serviceaccount/token +## AWS_REGION: us-east-1 +## +## 5. uncomment the EKS section in extraSecretMounts: below +## 6. uncomment the annotation section in the serviceAccount: above +## make sure to replace arn:aws:iam::123456789000:role/iam-role-name-here with your role arn + +env: {} + +## "valueFrom" environment variable references that will be added to deployment pods. Name is templated. +## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#envvarsource-v1-core +## Renders in container spec as: +## env: +## ... +## - name: +## valueFrom: +## +envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + +## The name of a secret in the same kubernetes namespace which contain values to be added to the environment +## This can be useful for auth tokens, etc. Value is templated. +envFromSecret: "" + +## Sensible environment variables that will be rendered as new secret object +## This can be useful for auth tokens, etc. +## If the secret values contains "{{", they'll need to be properly escaped so that they are not interpreted by Helm +## ref: https://helm.sh/docs/howto/charts_tips_and_tricks/#using-the-tpl-function +envRenderSecret: {} + +## The names of secrets in the same kubernetes namespace which contain values to be added to the environment +## Each entry should contain a name key, and can optionally specify whether the secret must be defined with an optional key. +## Name is templated. +envFromSecrets: [] +## - name: secret-name +## prefix: prefix +## optional: true + +## The names of conifgmaps in the same kubernetes namespace which contain values to be added to the environment +## Each entry should contain a name key, and can optionally specify whether the configmap must be defined with an optional key. +## Name is templated. +## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.23/#configmapenvsource-v1-core +envFromConfigMaps: [] +## - name: configmap-name +## prefix: prefix +## optional: true + +# Inject Kubernetes services as environment variables. +# See https://kubernetes.io/docs/concepts/services-networking/connect-applications-service/#environment-variables +enableServiceLinks: true + +## Additional grafana server secret mounts +# Defines additional mounts with secrets. Secrets must be manually created in the namespace. +extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # secretName: grafana-secret-files + # readOnly: true + # optional: false + # subPath: "" + # + # for AWS EKS (cloudwatch) use the following (see also instruction in env: above) + # - name: aws-iam-token + # mountPath: /var/run/secrets/eks.amazonaws.com/serviceaccount + # readOnly: true + # projected: + # defaultMode: 420 + # sources: + # - serviceAccountToken: + # audience: sts.amazonaws.com + # expirationSeconds: 86400 + # path: token + # + # for CSI e.g. Azure Key Vault use the following + # - name: secrets-store-inline + # mountPath: /run/secrets + # readOnly: true + # csi: + # driver: secrets-store.csi.k8s.io + # readOnly: true + # volumeAttributes: + # secretProviderClass: "akv-grafana-spc" + # nodePublishSecretRef: # Only required when using service principal mode + # name: grafana-akv-creds # Only required when using service principal mode + +## Additional grafana server volume mounts +# Defines additional volume mounts. +extraVolumeMounts: [] + # - name: extra-volume-0 + # mountPath: /mnt/volume0 + # readOnly: true + # - name: extra-volume-1 + # mountPath: /mnt/volume1 + # readOnly: true + # - name: grafana-secrets + # mountPath: /mnt/volume2 + +## Additional Grafana server volumes +extraVolumes: [] + # - name: extra-volume-0 + # existingClaim: volume-claim + # - name: extra-volume-1 + # hostPath: + # path: /usr/shared/ + # type: "" + # - name: grafana-secrets + # csi: + # driver: secrets-store.csi.k8s.io + # readOnly: true + # volumeAttributes: + # secretProviderClass: "grafana-env-spc" + +## Container Lifecycle Hooks. Execute a specific bash command or make an HTTP request +lifecycleHooks: {} + # postStart: + # exec: + # command: [] + +## Pass the plugins you want installed as a list. +## +plugins: [] + # - digrich-bubblechart-panel + # - grafana-clock-panel + ## You can also use other plugin download URL, as long as they are valid zip files, + ## and specify the name of the plugin after the semicolon. Like this: + # - https://grafana.com/api/plugins/marcusolsson-json-datasource/versions/1.3.2/download;marcusolsson-json-datasource + +## Configure grafana datasources +## ref: http://docs.grafana.org/administration/provisioning/#datasources +## +datasources: {} +# datasources.yaml: +# apiVersion: 1 +# datasources: +# - name: Prometheus +# type: prometheus +# url: http://prometheus-prometheus-server +# access: proxy +# isDefault: true +# - name: CloudWatch +# type: cloudwatch +# access: proxy +# uid: cloudwatch +# editable: false +# jsonData: +# authType: default +# defaultRegion: us-east-1 +# deleteDatasources: [] +# - name: Prometheus + +## Configure grafana alerting (can be templated) +## ref: http://docs.grafana.org/administration/provisioning/#alerting +## +alerting: {} + # rules.yaml: + # apiVersion: 1 + # groups: + # - orgId: 1 + # name: '{{ .Chart.Name }}_my_rule_group' + # folder: my_first_folder + # interval: 60s + # rules: + # - uid: my_id_1 + # title: my_first_rule + # condition: A + # data: + # - refId: A + # datasourceUid: '-100' + # model: + # conditions: + # - evaluator: + # params: + # - 3 + # type: gt + # operator: + # type: and + # query: + # params: + # - A + # reducer: + # type: last + # type: query + # datasource: + # type: __expr__ + # uid: '-100' + # expression: 1==0 + # intervalMs: 1000 + # maxDataPoints: 43200 + # refId: A + # type: math + # dashboardUid: my_dashboard + # panelId: 123 + # noDataState: Alerting + # for: 60s + # annotations: + # some_key: some_value + # labels: + # team: sre_team_1 + # contactpoints.yaml: + # secret: + # apiVersion: 1 + # contactPoints: + # - orgId: 1 + # name: cp_1 + # receivers: + # - uid: first_uid + # type: pagerduty + # settings: + # integrationKey: XXX + # severity: critical + # class: ping failure + # component: Grafana + # group: app-stack + # summary: | + # {{ `{{ include "default.message" . }}` }} + +## Configure notifiers +## ref: http://docs.grafana.org/administration/provisioning/#alert-notification-channels +## +notifiers: {} +# notifiers.yaml: +# notifiers: +# - name: email-notifier +# type: email +# uid: email1 +# # either: +# org_id: 1 +# # or +# org_name: Main Org. +# is_default: true +# settings: +# addresses: an_email_address@example.com +# delete_notifiers: + +## Configure grafana dashboard providers +## ref: http://docs.grafana.org/administration/provisioning/#dashboards +## +## `path` must be /var/lib/grafana/dashboards/ +## +dashboardProviders: {} +# dashboardproviders.yaml: +# apiVersion: 1 +# providers: +# - name: 'default' +# orgId: 1 +# folder: '' +# type: file +# disableDeletion: false +# editable: true +# options: +# path: /var/lib/grafana/dashboards/default + +## Configure grafana dashboard to import +## NOTE: To use dashboards you must also enable/configure dashboardProviders +## ref: https://grafana.com/dashboards +## +## dashboards per provider, use provider name as key. +## +dashboards: {} + # default: + # some-dashboard: + # json: | + # $RAW_JSON + # custom-dashboard: + # file: dashboards/custom-dashboard.json + # prometheus-stats: + # gnetId: 2 + # revision: 2 + # datasource: Prometheus + # local-dashboard: + # url: https://example.com/repository/test.json + # token: '' + # local-dashboard-base64: + # url: https://example.com/repository/test-b64.json + # token: '' + # b64content: true + # local-dashboard-gitlab: + # url: https://example.com/repository/test-gitlab.json + # gitlabToken: '' + # local-dashboard-bitbucket: + # url: https://example.com/repository/test-bitbucket.json + # bearerToken: '' + # local-dashboard-azure: + # url: https://example.com/repository/test-azure.json + # basic: '' + # acceptHeader: '*/*' + +## Reference to external ConfigMap per provider. Use provider name as key and ConfigMap name as value. +## A provider dashboards must be defined either by external ConfigMaps or in values.yaml, not in both. +## ConfigMap data example: +## +## data: +## example-dashboard.json: | +## RAW_JSON +## +dashboardsConfigMaps: {} +# default: "" + +## Grafana's primary configuration +## NOTE: values in map will be converted to ini format +## ref: http://docs.grafana.org/installation/configuration/ +## +grafana.ini: + paths: + data: /var/lib/grafana/ + logs: /var/log/grafana + plugins: /var/lib/grafana/plugins + provisioning: /etc/grafana/provisioning + analytics: + check_for_updates: true + log: + mode: console + grafana_net: + url: https://grafana.net + server: + domain: "{{ if (and .Values.ingress.enabled .Values.ingress.hosts) }}{{ tpl (.Values.ingress.hosts | first) . }}{{ else }}''{{ end }}" +## grafana Authentication can be enabled with the following values on grafana.ini + # server: + # The full public facing url you use in browser, used for redirects and emails + # root_url: + # https://grafana.com/docs/grafana/latest/auth/github/#enable-github-in-grafana + # auth.github: + # enabled: false + # allow_sign_up: false + # scopes: user:email,read:org + # auth_url: https://github.com/login/oauth/authorize + # token_url: https://github.com/login/oauth/access_token + # api_url: https://api.github.com/user + # team_ids: + # allowed_organizations: + # client_id: + # client_secret: +## LDAP Authentication can be enabled with the following values on grafana.ini +## NOTE: Grafana will fail to start if the value for ldap.toml is invalid + # auth.ldap: + # enabled: true + # allow_sign_up: true + # config_file: /etc/grafana/ldap.toml + +## Grafana's LDAP configuration +## Templated by the template in _helpers.tpl +## NOTE: To enable the grafana.ini must be configured with auth.ldap.enabled +## ref: http://docs.grafana.org/installation/configuration/#auth-ldap +## ref: http://docs.grafana.org/installation/ldap/#configuration +ldap: + enabled: false + # `existingSecret` is a reference to an existing secret containing the ldap configuration + # for Grafana in a key `ldap-toml`. + existingSecret: "" + # `config` is the content of `ldap.toml` that will be stored in the created secret + config: "" + # config: |- + # verbose_logging = true + + # [[servers]] + # host = "my-ldap-server" + # port = 636 + # use_ssl = true + # start_tls = false + # ssl_skip_verify = false + # bind_dn = "uid=%s,ou=users,dc=myorg,dc=com" + +## Grafana's SMTP configuration +## NOTE: To enable, grafana.ini must be configured with smtp.enabled +## ref: http://docs.grafana.org/installation/configuration/#smtp +smtp: + # `existingSecret` is a reference to an existing secret containing the smtp configuration + # for Grafana. + existingSecret: "" + userKey: "user" + passwordKey: "password" + +## Sidecars that collect the configmaps with specified label and stores the included files them into the respective folders +## Requires at least Grafana 5 to work and can't be used together with parameters dashboardProviders, datasources and dashboards +sidecar: + image: + # -- The Docker registry + registry: quay.io + repository: kiwigrid/k8s-sidecar + tag: 1.28.0 + sha: "" + imagePullPolicy: IfNotPresent + resources: {} +# limits: +# cpu: 100m +# memory: 100Mi +# requests: +# cpu: 50m +# memory: 50Mi + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + seccompProfile: + type: RuntimeDefault + # skipTlsVerify Set to true to skip tls verification for kube api calls + # skipTlsVerify: true + enableUniqueFilenames: false + readinessProbe: {} + livenessProbe: {} + # Log level default for all sidecars. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. Defaults to INFO + # logLevel: INFO + alerts: + enabled: false + # Additional environment variables for the alerts sidecar + env: {} + # Do not reprocess already processed unchanged resources on k8s API reconnect. + # ignoreAlreadyProcessed: true + # label that the configmaps with alert are marked with + label: grafana_alert + # value of label that the configmaps with alert are set to + labelValue: "" + # Log level. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. + # logLevel: INFO + # If specified, the sidecar will search for alert config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # Endpoint to send request to reload alerts + reloadURL: "http://localhost:3000/api/admin/provisioning/alerting/reload" + # Absolute path to shell script to execute after a alert got reloaded + script: null + skipReload: false + # This is needed if skipReload is true, to load any alerts defined at startup time. + # Deploy the alert sidecar as an initContainer. + initAlerts: false + # Additional alerts sidecar volume mounts + extraMounts: [] + # Sets the size limit of the alert sidecar emptyDir volume + sizeLimit: {} + dashboards: + enabled: false + # Additional environment variables for the dashboards sidecar + env: {} + ## "valueFrom" environment variable references that will be added to deployment pods. Name is templated. + ## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#envvarsource-v1-core + ## Renders in container spec as: + ## env: + ## ... + ## - name: + ## valueFrom: + ## + envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + # Do not reprocess already processed unchanged resources on k8s API reconnect. + # ignoreAlreadyProcessed: true + SCProvider: true + # label that the configmaps with dashboards are marked with + label: grafana_dashboard + # value of label that the configmaps with dashboards are set to + labelValue: "" + # Log level. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. + # logLevel: INFO + # folder in the pod that should hold the collected dashboards (unless `defaultFolderName` is set) + folder: /tmp/dashboards + # The default folder name, it will create a subfolder under the `folder` and put dashboards in there instead + defaultFolderName: null + # Namespaces list. If specified, the sidecar will search for config-maps/secrets inside these namespaces. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces. + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # If specified, the sidecar will look for annotation with this name to create folder and put graph here. + # You can use this parameter together with `provider.foldersFromFilesStructure`to annotate configmaps and create folder structure. + folderAnnotation: null + # Endpoint to send request to reload alerts + reloadURL: "http://localhost:3000/api/admin/provisioning/dashboards/reload" + # Absolute path to shell script to execute after a configmap got reloaded + script: null + skipReload: false + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # provider configuration that lets grafana manage the dashboards + provider: + # name of the provider, should be unique + name: sidecarProvider + # orgid as configured in grafana + orgid: 1 + # folder in which the dashboards should be imported in grafana + folder: '' + # folder UID. will be automatically generated if not specified + folderUid: '' + # type of the provider + type: file + # disableDelete to activate a import-only behaviour + disableDelete: false + # allow updating provisioned dashboards from the UI + allowUiUpdates: false + # allow Grafana to replicate dashboard structure from filesystem + foldersFromFilesStructure: false + # Additional dashboards sidecar volume mounts + extraMounts: [] + # Sets the size limit of the dashboard sidecar emptyDir volume + sizeLimit: {} + datasources: + enabled: false + # Additional environment variables for the datasourcessidecar + env: {} + ## "valueFrom" environment variable references that will be added to deployment pods. Name is templated. + ## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#envvarsource-v1-core + ## Renders in container spec as: + ## env: + ## ... + ## - name: + ## valueFrom: + ## + envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + # Do not reprocess already processed unchanged resources on k8s API reconnect. + # ignoreAlreadyProcessed: true + # label that the configmaps with datasources are marked with + label: grafana_datasource + # value of label that the configmaps with datasources are set to + labelValue: "" + # Log level. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. + # logLevel: INFO + # If specified, the sidecar will search for datasource config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # Endpoint to send request to reload datasources + reloadURL: "http://localhost:3000/api/admin/provisioning/datasources/reload" + # Absolute path to shell script to execute after a datasource got reloaded + script: null + skipReload: false + # This is needed if skipReload is true, to load any datasources defined at startup time. + # Deploy the datasources sidecar as an initContainer. + initDatasources: false + # Additional datasources sidecar volume mounts + extraMounts: [] + # Sets the size limit of the datasource sidecar emptyDir volume + sizeLimit: {} + plugins: + enabled: false + # Additional environment variables for the plugins sidecar + env: {} + # Do not reprocess already processed unchanged resources on k8s API reconnect. + # ignoreAlreadyProcessed: true + # label that the configmaps with plugins are marked with + label: grafana_plugin + # value of label that the configmaps with plugins are set to + labelValue: "" + # Log level. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. + # logLevel: INFO + # If specified, the sidecar will search for plugin config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # Endpoint to send request to reload plugins + reloadURL: "http://localhost:3000/api/admin/provisioning/plugins/reload" + # Absolute path to shell script to execute after a plugin got reloaded + script: null + skipReload: false + # Deploy the datasource sidecar as an initContainer in addition to a container. + # This is needed if skipReload is true, to load any plugins defined at startup time. + initPlugins: false + # Additional plugins sidecar volume mounts + extraMounts: [] + # Sets the size limit of the plugin sidecar emptyDir volume + sizeLimit: {} + notifiers: + enabled: false + # Additional environment variables for the notifierssidecar + env: {} + # Do not reprocess already processed unchanged resources on k8s API reconnect. + # ignoreAlreadyProcessed: true + # label that the configmaps with notifiers are marked with + label: grafana_notifier + # value of label that the configmaps with notifiers are set to + labelValue: "" + # Log level. Can be one of: DEBUG, INFO, WARN, ERROR, CRITICAL. + # logLevel: INFO + # If specified, the sidecar will search for notifier config-maps inside this namespace. + # Otherwise the namespace in which the sidecar is running will be used. + # It's also possible to specify ALL to search in all namespaces + searchNamespace: null + # Method to use to detect ConfigMap changes. With WATCH the sidecar will do a WATCH requests, with SLEEP it will list all ConfigMaps, then sleep for 60 seconds. + watchMethod: WATCH + # search in configmap, secret or both + resource: both + # watchServerTimeout: request to the server, asking it to cleanly close the connection after that. + # defaults to 60sec; much higher values like 3600 seconds (1h) are feasible for non-Azure K8S + # watchServerTimeout: 3600 + # + # watchClientTimeout: is a client-side timeout, configuring your local socket. + # If you have a network outage dropping all packets with no RST/FIN, + # this is how long your client waits before realizing & dropping the connection. + # defaults to 66sec (sic!) + # watchClientTimeout: 60 + # + # Endpoint to send request to reload notifiers + reloadURL: "http://localhost:3000/api/admin/provisioning/notifications/reload" + # Absolute path to shell script to execute after a notifier got reloaded + script: null + skipReload: false + # Deploy the notifier sidecar as an initContainer in addition to a container. + # This is needed if skipReload is true, to load any notifiers defined at startup time. + initNotifiers: false + # Additional notifiers sidecar volume mounts + extraMounts: [] + # Sets the size limit of the notifier sidecar emptyDir volume + sizeLimit: {} + +## Override the deployment namespace +## +namespaceOverride: "" + +## Number of old ReplicaSets to retain +## +revisionHistoryLimit: 10 + +## Add a seperate remote image renderer deployment/service +imageRenderer: + deploymentStrategy: {} + # Enable the image-renderer deployment & service + enabled: false + replicas: 1 + autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 5 + targetCPU: "60" + targetMemory: "" + behavior: {} + # The url of remote image renderer if it is not in the same namespace with the grafana instance + serverURL: "" + # The callback url of grafana instances if it is not in the same namespace with the remote image renderer + renderingCallbackURL: "" + image: + # -- The Docker registry + registry: docker.io + # image-renderer Image repository + repository: grafana/grafana-image-renderer + # image-renderer Image tag + tag: latest + # image-renderer Image sha (optional) + sha: "" + # image-renderer ImagePullPolicy + pullPolicy: Always + # extra environment variables + env: + HTTP_HOST: "0.0.0.0" + # RENDERING_ARGS: --no-sandbox,--disable-gpu,--window-size=1280x758 + # RENDERING_MODE: clustered + # IGNORE_HTTPS_ERRORS: true + + ## "valueFrom" environment variable references that will be added to deployment pods. Name is templated. + ## ref: https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.19/#envvarsource-v1-core + ## Renders in container spec as: + ## env: + ## ... + ## - name: + ## valueFrom: + ## + envValueFrom: {} + # ENV_NAME: + # configMapKeyRef: + # name: configmap-name + # key: value_key + + # image-renderer deployment serviceAccount + serviceAccountName: "" + # image-renderer deployment securityContext + securityContext: {} + # image-renderer deployment container securityContext + containerSecurityContext: + seccompProfile: + type: RuntimeDefault + capabilities: + drop: ['ALL'] + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + ## image-renderer pod annotation + podAnnotations: {} + # image-renderer deployment Host Aliases + hostAliases: [] + # image-renderer deployment priority class + priorityClassName: '' + service: + # Enable the image-renderer service + enabled: true + # image-renderer service port name + portName: 'http' + # image-renderer service port used by both service and deployment + port: 8081 + targetPort: 8081 + # Adds the appProtocol field to the image-renderer service. This allows to work with istio protocol selection. Ex: "http" or "tcp" + appProtocol: "" + serviceMonitor: + ## If true, a ServiceMonitor CRD is created for a prometheus operator + ## https://github.com/coreos/prometheus-operator + ## + enabled: false + path: /metrics + # namespace: monitoring (defaults to use the namespace this chart is deployed to) + labels: {} + interval: 1m + scheme: http + tlsConfig: {} + scrapeTimeout: 30s + relabelings: [] + # See: https://doc.crds.dev/github.com/prometheus-operator/kube-prometheus/monitoring.coreos.com/ServiceMonitor/v1@v0.11.0#spec-targetLabels + targetLabels: [] + # - targetLabel1 + # - targetLabel2 + # If https is enabled in Grafana, this needs to be set as 'https' to correctly configure the callback used in Grafana + grafanaProtocol: http + # In case a sub_path is used this needs to be added to the image renderer callback + grafanaSubPath: "" + # name of the image-renderer port on the pod + podPortName: http + # number of image-renderer replica sets to keep + revisionHistoryLimit: 10 + networkPolicy: + # Enable a NetworkPolicy to limit inbound traffic to only the created grafana pods + limitIngress: true + # Enable a NetworkPolicy to limit outbound traffic to only the created grafana pods + limitEgress: false + # Allow additional services to access image-renderer (eg. Prometheus operator when ServiceMonitor is enabled) + extraIngressSelectors: [] + resources: {} +# limits: +# cpu: 100m +# memory: 100Mi +# requests: +# cpu: 50m +# memory: 50Mi + ## Node labels for pod assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + # + nodeSelector: {} + + ## Tolerations for pod assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + + ## Affinity for pod assignment (evaluated as template) + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## + affinity: {} + + ## Use an alternate scheduler, e.g. "stork". + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + ## + # schedulerName: "default-scheduler" + + # Extra configmaps to mount in image-renderer pods + extraConfigmapMounts: [] + + # Extra secrets to mount in image-renderer pods + extraSecretMounts: [] + + # Extra volumes to mount in image-renderer pods + extraVolumeMounts: [] + + # Extra volumes for image-renderer pods + extraVolumes: [] + +networkPolicy: + ## @param networkPolicy.enabled Enable creation of NetworkPolicy resources. Only Ingress traffic is filtered for now. + ## + enabled: false + ## @param networkPolicy.allowExternal Don't require client label for connections + ## The Policy model to apply. When set to false, only pods with the correct + ## client label will have network access to grafana port defined. + ## When true, grafana will accept connections from any source + ## (with the correct destination port). + ## + ingress: true + ## @param networkPolicy.ingress When true enables the creation + ## an ingress network policy + ## + allowExternal: true + ## @param networkPolicy.explicitNamespacesSelector A Kubernetes LabelSelector to explicitly select namespaces from which traffic could be allowed + ## If explicitNamespacesSelector is missing or set to {}, only client Pods that are in the networkPolicy's namespace + ## and that match other criteria, the ones that have the good label, can reach the grafana. + ## But sometimes, we want the grafana to be accessible to clients from other namespaces, in this case, we can use this + ## LabelSelector to select these namespaces, note that the networkPolicy's namespace should also be explicitly added. + ## + ## Example: + ## explicitNamespacesSelector: + ## matchLabels: + ## role: frontend + ## matchExpressions: + ## - {key: role, operator: In, values: [frontend]} + ## + explicitNamespacesSelector: {} + ## + ## + ## + ## + ## + ## + egress: + ## @param networkPolicy.egress.enabled When enabled, an egress network policy will be + ## created allowing grafana to connect to external data sources from kubernetes cluster. + enabled: false + ## + ## @param networkPolicy.egress.blockDNSResolution When enabled, DNS resolution will be blocked + ## for all pods in the grafana namespace. + blockDNSResolution: false + ## + ## @param networkPolicy.egress.ports Add individual ports to be allowed by the egress + ports: [] + ## Add ports to the egress by specifying - port: + ## E.X. + ## - port: 80 + ## - port: 443 + ## + ## @param networkPolicy.egress.to Allow egress traffic to specific destinations + to: [] + ## Add destinations to the egress by specifying - ipBlock: + ## E.X. + ## to: + ## - namespaceSelector: + ## matchExpressions: + ## - {key: role, operator: In, values: [grafana]} + ## + ## + ## + ## + ## + +# Enable backward compatibility of kubernetes where version below 1.13 doesn't have the enableServiceLinks option +enableKubeBackwardCompatibility: false +useStatefulSet: false +# Create a dynamic manifests via values: +extraObjects: [] + # - apiVersion: "kubernetes-client.io/v1" + # kind: ExternalSecret + # metadata: + # name: grafana-secrets + # spec: + # backendType: gcpSecretsManager + # data: + # - key: grafana-admin-password + # name: adminPassword + +# assertNoLeakedSecrets is a helper function defined in _helpers.tpl that checks if secret +# values are not exposed in the rendered grafana.ini configmap. It is enabled by default. +# +# To pass values into grafana.ini without exposing them in a configmap, use variable expansion: +# https://grafana.com/docs/grafana/latest/setup-grafana/configure-grafana/#variable-expansion +# +# Alternatively, if you wish to allow secret values to be exposed in the rendered grafana.ini configmap, +# you can disable this check by setting assertNoLeakedSecrets to false. +assertNoLeakedSecrets: true diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/.helmignore b/charts/kasten/k10/7.0.1401/charts/prometheus/.helmignore new file mode 100644 index 000000000..825c00779 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/.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 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.lock b/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.lock new file mode 100644 index 000000000..557cf7ebf --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.lock @@ -0,0 +1,15 @@ +dependencies: +- name: alertmanager + repository: https://prometheus-community.github.io/helm-charts + version: 1.13.0 +- name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.26.0 +- name: prometheus-node-exporter + repository: https://prometheus-community.github.io/helm-charts + version: 4.40.0 +- name: prometheus-pushgateway + repository: https://prometheus-community.github.io/helm-charts + version: 2.15.0 +digest: "" +generated: "0001-01-01T00:00:00Z" diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.yaml new file mode 100644 index 000000000..779a3b594 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/Chart.yaml @@ -0,0 +1,53 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/prometheus-community/helm-charts + - name: Upstream Project + url: https://github.com/prometheus/prometheus +apiVersion: v2 +appVersion: v2.55.0 +dependencies: +- condition: alertmanager.enabled + name: alertmanager + repository: https://prometheus-community.github.io/helm-charts + version: 1.13.* +- condition: kube-state-metrics.enabled + name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.26.* +- condition: prometheus-node-exporter.enabled + name: prometheus-node-exporter + repository: https://prometheus-community.github.io/helm-charts + version: 4.40.* +- condition: prometheus-pushgateway.enabled + name: prometheus-pushgateway + repository: https://prometheus-community.github.io/helm-charts + version: 2.15.* +description: Prometheus is a monitoring system and time series database. +home: https://prometheus.io/ +icon: https://raw.githubusercontent.com/prometheus/prometheus.github.io/master/assets/prometheus_logo-cb55bb5c346.png +keywords: +- monitoring +- prometheus +kubeVersion: '>=1.19.0-0' +maintainers: +- email: gianrubio@gmail.com + name: gianrubio +- email: zanhsieh@gmail.com + name: zanhsieh +- email: miroslav.hadzhiev@gmail.com + name: Xtigyro +- email: naseem@transit.app + name: naseemkullah +- email: rootsandtrees@posteo.de + name: zeritti +name: prometheus +sources: +- https://github.com/prometheus/alertmanager +- https://github.com/prometheus/prometheus +- https://github.com/prometheus/pushgateway +- https://github.com/prometheus/node_exporter +- https://github.com/kubernetes/kube-state-metrics +type: application +version: 25.28.0 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/OWNERS b/charts/kasten/k10/7.0.1401/charts/prometheus/OWNERS new file mode 100644 index 000000000..0cfd95021 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/OWNERS @@ -0,0 +1,6 @@ +approvers: +- mgoodness +- gianrubio +reviewers: +- mgoodness +- gianrubio diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/README.md b/charts/kasten/k10/7.0.1401/charts/prometheus/README.md new file mode 100644 index 000000000..2cb744ce8 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/README.md @@ -0,0 +1,382 @@ +# Prometheus + +[Prometheus](https://prometheus.io/), a [Cloud Native Computing Foundation](https://cncf.io/) project, is a systems and service monitoring system. It collects metrics from configured targets at given intervals, evaluates rule expressions, displays the results, and can trigger alerts if some condition is observed to be true. + +This chart bootstraps a [Prometheus](https://prometheus.io/) deployment on a [Kubernetes](http://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. + +## Prerequisites + +- Kubernetes 1.19+ +- Helm 3.7+ + +## Get Repository Info + +```console +helm repo add prometheus-community https://prometheus-community.github.io/helm-charts +helm repo update +``` + +_See [helm repository](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Install Chart + +Starting with version 16.0, the Prometheus chart requires Helm 3.7+ in order to install successfully. Please check your `helm` release before installation. + +```console +helm install [RELEASE_NAME] prometheus-community/prometheus +``` + +_See [configuration](#configuration) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Dependencies + +By default this chart installs additional, dependent charts: + +- [alertmanager](https://github.com/prometheus-community/helm-charts/tree/main/charts/alertmanager) +- [kube-state-metrics](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics) +- [prometheus-node-exporter](https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-node-exporter) +- [prometheus-pushgateway](https://github.com/walker-tom/helm-charts/tree/main/charts/prometheus-pushgateway) + +To disable the dependency during installation, set `alertmanager.enabled`, `kube-state-metrics.enabled`, `prometheus-node-exporter.enabled` and `prometheus-pushgateway.enabled` to `false`. + +_See [helm dependency](https://helm.sh/docs/helm/helm_dependency/) for command documentation._ + +## Uninstall Chart + +```console +helm uninstall [RELEASE_NAME] +``` + +This removes all the Kubernetes components associated with the chart and deletes the release. + +_See [helm uninstall](https://helm.sh/docs/helm/helm_uninstall/) for command documentation._ + +## Updating values.schema.json + +A [`values.schema.json`](https://helm.sh/docs/topics/charts/#schema-files) file has been added to validate chart values. When `values.yaml` file has a structure change (i.e. add a new field, change value type, etc.), modify `values.schema.json` file manually or run `helm schema-gen values.yaml > values.schema.json` to ensure the schema is aligned with the latest values. Refer to [helm plugin `helm-schema-gen`](https://github.com/karuppiah7890/helm-schema-gen) for plugin installation instructions. + +## Upgrading Chart + +```console +helm upgrade [RELEASE_NAME] prometheus-community/prometheus --install +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +### To 25.0 + +The `server.remoteRead[].url` and `server.remoteWrite[].url` fields now support templating. Allowing for `url` values such as `https://{{ .Release.Name }}.example.com`. + +Any entries in these which previously included `{{` or `}}` must be escaped with `{{ "{{" }}` and `{{ "}}" }}` respectively. Entries which did not previously include the template-like syntax will not be affected. + +### To 24.0 + +Require Kubernetes 1.19+ + +Release 1.0.0 of the _alertmanager_ replaced [configmap-reload](https://github.com/jimmidyson/configmap-reload) with [prometheus-config-reloader](https://github.com/prometheus-operator/prometheus-operator/tree/main/cmd/prometheus-config-reloader). +Extra command-line arguments specified via `configmapReload.prometheus.extraArgs` are not compatible and will break with the new prometheus-config-reloader. Please, refer to the [sources](https://github.com/prometheus-operator/prometheus-operator/blob/main/cmd/prometheus-config-reloader/main.go) in order to make the appropriate adjustment to the extra command-line arguments. + +### To 23.0 + +Release 5.0.0 of the _kube-state-metrics_ chart introduced a separation of the `image.repository` value in two distinct values: + +```console + image: + registry: registry.k8s.io + repository: kube-state-metrics/kube-state-metrics +``` + +If a custom values file or CLI flags set `kube-state.metrics.image.repository`, please, set the new values accordingly. + +If you are upgrading _prometheus-pushgateway_ with the chart and _prometheus-pushgateway_ has been deployed as a statefulset with a persistent volume, the statefulset must be deleted before upgrading the chart, e.g.: + +```bash +kubectl delete sts -l app.kubernetes.io/name=prometheus-pushgateway -n monitoring --cascade=orphan +``` + +Users are advised to review changes in the corresponding chart releases before upgrading. + +### To 22.0 + +The `app.kubernetes.io/version` label has been removed from the pod selector. + +Therefore, you must delete the previous StatefulSet or Deployment before upgrading. Performing this operation will cause **Prometheus to stop functioning** until the upgrade is complete. + +```console +kubectl delete deploy,sts -l app.kubernetes.io/name=prometheus +``` + +### To 21.0 + +The Kubernetes labels have been updated to follow [Helm 3 label and annotation best practices](https://helm.sh/docs/chart_best_practices/labels/). +Specifically, labels mapping is listed below: + +| OLD | NEW | +|--------------------|------------------------------| +|heritage | app.kubernetes.io/managed-by | +|chart | helm.sh/chart | +|[container version] | app.kubernetes.io/version | +|app | app.kubernetes.io/name | +|release | app.kubernetes.io/instance | + +Therefore, depending on the way you've configured the chart, the previous StatefulSet or Deployment need to be deleted before upgrade. + +If `runAsStatefulSet: false` (this is the default): + +```console +kubectl delete deploy -l app=prometheus +``` + +If `runAsStatefulSet: true`: + +```console +kubectl delete sts -l app=prometheus +``` + +After that do the actual upgrade: + +```console +helm upgrade -i prometheus prometheus-community/prometheus +``` + +### To 20.0 + +The [configmap-reload](https://github.com/jimmidyson/configmap-reload) container was replaced by the [prometheus-config-reloader](https://github.com/prometheus-operator/prometheus-operator/tree/main/cmd/prometheus-config-reloader). +Extra command-line arguments specified via configmapReload.prometheus.extraArgs are not compatible and will break with the new prometheus-config-reloader, refer to the [sources](https://github.com/prometheus-operator/prometheus-operator/blob/main/cmd/prometheus-config-reloader/main.go) in order to make the appropriate adjustment to the extra command-line arguments. + +### To 19.0 + +Prometheus has been updated to version v2.40.5. + +Prometheus-pushgateway was updated to version 2.0.0 which adapted [Helm label and annotation best practices](https://helm.sh/docs/chart_best_practices/labels/). +See the [upgrade docs of the prometheus-pushgateway chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-pushgateway#to-200) to see whats to do, before you upgrade Prometheus! + +The condition in Chart.yaml to disable kube-state-metrics has been changed from `kubeStateMetrics.enabled` to `kube-state-metrics.enabled` + +The Docker image tag is used from appVersion field in Chart.yaml by default. + +Unused subchart configs has been removed and subchart config is now on the bottom of the config file. + +If Prometheus is used as deployment the updatestrategy has been changed to "Recreate" by default, so Helm updates work out of the box. + +`.Values.server.extraTemplates` & `.Values.server.extraObjects` has been removed in favour of `.Values.extraManifests`, which can do the same. + +`.Values.server.enabled` has been removed as it's useless now that all components are created by subcharts. + +All files in `templates/server` directory has been moved to `templates` directory. + +```bash +helm upgrade [RELEASE_NAME] prometheus-community/prometheus --version 19.0.0 +``` + +### To 18.0 + +Version 18.0.0 uses alertmanager service from the [alertmanager chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/alertmanager). If you've made some config changes, please check the old `alertmanager` and the new `alertmanager` configuration section in values.yaml for differences. + +Note that the `configmapReload` section for `alertmanager` was moved out of dedicated section (`configmapReload.alertmanager`) to alertmanager embedded (`alertmanager.configmapReload`). + +Before you update, please scale down the `prometheus-server` deployment to `0` then perform upgrade: + +```bash +# In 17.x +kubectl scale deploy prometheus-server --replicas=0 +# Upgrade +helm upgrade [RELEASE_NAME] prometheus-community/prometheus --version 18.0.0 +``` + +### To 17.0 + +Version 17.0.0 uses pushgateway service from the [prometheus-pushgateway chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-pushgateway). If you've made some config changes, please check the old `pushgateway` and the new `prometheus-pushgateway` configuration section in values.yaml for differences. + +Before you update, please scale down the `prometheus-server` deployment to `0` then perform upgrade: + +```bash +# In 16.x +kubectl scale deploy prometheus-server --replicas=0 +# Upgrade +helm upgrade [RELEASE_NAME] prometheus-community/prometheus --version 17.0.0 +``` + +### To 16.0 + +Starting from version 16.0 embedded services (like alertmanager, node-exporter etc.) are moved out of Prometheus chart and the respecting charts from this repository are used as dependencies. Version 16.0.0 moves node-exporter service to [prometheus-node-exporter chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-node-exporter). If you've made some config changes, please check the old `nodeExporter` and the new `prometheus-node-exporter` configuration section in values.yaml for differences. + +Before you update, please scale down the `prometheus-server` deployment to `0` then perform upgrade: + +```bash +# In 15.x +kubectl scale deploy prometheus-server --replicas=0 +# Upgrade +helm upgrade [RELEASE_NAME] prometheus-community/prometheus --version 16.0.0 +``` + +### To 15.0 + +Version 15.0.0 changes the relabeling config, aligning it with the [Prometheus community conventions](https://github.com/prometheus/prometheus/pull/9832). If you've made manual changes to the relabeling config, you have to adapt your changes. + +Before you update please execute the following command, to be able to update kube-state-metrics: + +```bash +kubectl delete deployments.apps -l app.kubernetes.io/instance=prometheus,app.kubernetes.io/name=kube-state-metrics --cascade=orphan +``` + +### To 9.0 + +Version 9.0 adds a new option to enable or disable the Prometheus Server. This supports the use case of running a Prometheus server in one k8s cluster and scraping exporters in another cluster while using the same chart for each deployment. To install the server `server.enabled` must be set to `true`. + +### To 5.0 + +As of version 5.0, this chart uses Prometheus 2.x. This version of prometheus introduces a new data format and is not compatible with prometheus 1.x. It is recommended to install this as a new release, as updating existing releases will not work. See the [prometheus docs](https://prometheus.io/docs/prometheus/latest/migration/#storage) for instructions on retaining your old data. + +Prometheus version 2.x has made changes to alertmanager, storage and recording rules. Check out the migration guide [here](https://prometheus.io/docs/prometheus/2.0/migration/). + +Users of this chart will need to update their alerting rules to the new format before they can upgrade. + +### Example Migration + +Assuming you have an existing release of the prometheus chart, named `prometheus-old`. In order to update to prometheus 2.x while keeping your old data do the following: + +1. Update the `prometheus-old` release. Disable scraping on every component besides the prometheus server, similar to the configuration below: + + ```yaml + alertmanager: + enabled: false + alertmanagerFiles: + alertmanager.yml: "" + kubeStateMetrics: + enabled: false + nodeExporter: + enabled: false + pushgateway: + enabled: false + server: + extraArgs: + storage.local.retention: 720h + serverFiles: + alerts: "" + prometheus.yml: "" + rules: "" + ``` + +1. Deploy a new release of the chart with version 5.0+ using prometheus 2.x. In the values.yaml set the scrape config as usual, and also add the `prometheus-old` instance as a remote-read target. + + ```yaml + prometheus.yml: + ... + remote_read: + - url: http://prometheus-old/api/v1/read + ... + ``` + + Old data will be available when you query the new prometheus instance. + +## Configuration + +See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). To see all configurable options with detailed comments, visit the chart's [values.yaml](./values.yaml), or run these configuration commands: + +```console +helm show values prometheus-community/prometheus +``` + +You may similarly use the above configuration commands on each chart [dependency](#dependencies) to see its configurations. + +### Scraping Pod Metrics via Annotations + +This chart uses a default configuration that causes prometheus to scrape a variety of kubernetes resource types, provided they have the correct annotations. In this section we describe how to configure pods to be scraped; for information on how other resource types can be scraped you can do a `helm template` to get the kubernetes resource definitions, and then reference the prometheus configuration in the ConfigMap against the prometheus documentation for [relabel_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config) and [kubernetes_sd_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#kubernetes_sd_config). + +In order to get prometheus to scrape pods, you must add annotations to the pods as below: + +```yaml +metadata: + annotations: + prometheus.io/scrape: "true" + prometheus.io/path: /metrics + prometheus.io/port: "8080" +``` + +You should adjust `prometheus.io/path` based on the URL that your pod serves metrics from. `prometheus.io/port` should be set to the port that your pod serves metrics from. Note that the values for `prometheus.io/scrape` and `prometheus.io/port` must be enclosed in double quotes. + +### Sharing Alerts Between Services + +Note that when [installing](#install-chart) or [upgrading](#upgrading-chart) you may use multiple values override files. This is particularly useful when you have alerts belonging to multiple services in the cluster. For example, + +```yaml +# values.yaml +# ... + +# service1-alert.yaml +serverFiles: + alerts: + service1: + - alert: anAlert + # ... + +# service2-alert.yaml +serverFiles: + alerts: + service2: + - alert: anAlert + # ... +``` + +```console +helm install [RELEASE_NAME] prometheus-community/prometheus -f values.yaml -f service1-alert.yaml -f service2-alert.yaml +``` + +### RBAC Configuration + +Roles and RoleBindings resources will be created automatically for `server` service. + +To manually setup RBAC you need to set the parameter `rbac.create=false` and specify the service account to be used for each service by setting the parameters: `serviceAccounts.{{ component }}.create` to `false` and `serviceAccounts.{{ component }}.name` to the name of a pre-existing service account. + +> **Tip**: You can refer to the default `*-clusterrole.yaml` and `*-clusterrolebinding.yaml` files in [templates](templates/) to customize your own. + +### ConfigMap Files + +AlertManager is configured through [alertmanager.yml](https://prometheus.io/docs/alerting/configuration/). This file (and any others listed in `alertmanagerFiles`) will be mounted into the `alertmanager` pod. + +Prometheus is configured through [prometheus.yml](https://prometheus.io/docs/operating/configuration/). This file (and any others listed in `serverFiles`) will be mounted into the `server` pod. + +### Ingress TLS + +If your cluster allows automatic creation/retrieval of TLS certificates (e.g. [cert-manager](https://github.com/jetstack/cert-manager)), please refer to the documentation for that mechanism. + +To manually configure TLS, first create/retrieve a key & certificate pair for the address(es) you wish to protect. Then create a TLS secret in the namespace: + +```console +kubectl create secret tls prometheus-server-tls --cert=path/to/tls.cert --key=path/to/tls.key +``` + +Include the secret's name, along with the desired hostnames, in the alertmanager/server Ingress TLS section of your custom `values.yaml` file: + +```yaml +server: + ingress: + ## If true, Prometheus server Ingress will be created + ## + enabled: true + + ## Prometheus server Ingress hostnames + ## Must be provided if Ingress is enabled + ## + hosts: + - prometheus.domain.com + + ## Prometheus server Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: + - secretName: prometheus-server-tls + hosts: + - prometheus.domain.com +``` + +### NetworkPolicy + +Enabling Network Policy for Prometheus will secure connections to Alert Manager and Kube State Metrics by only accepting connections from Prometheus Server. All inbound connections to Prometheus Server are still allowed. + +To enable network policy for Prometheus, install a networking plugin that implements the Kubernetes NetworkPolicy spec, and set `networkPolicy.enabled` to true. + +If NetworkPolicy is enabled for Prometheus' scrape targets, you may also need to manually create a networkpolicy which allows it. diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/.helmignore b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/.helmignore new file mode 100644 index 000000000..7653e97e6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/.helmignore @@ -0,0 +1,25 @@ +# 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 +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +unittests/ diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/Chart.yaml new file mode 100644 index 000000000..bf2bfa160 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/Chart.yaml @@ -0,0 +1,24 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/prometheus-community/helm-charts +apiVersion: v2 +appVersion: v0.27.0 +description: The Alertmanager handles alerts sent by client applications such as the + Prometheus server. +home: https://prometheus.io/ +icon: https://raw.githubusercontent.com/prometheus/prometheus.github.io/master/assets/prometheus_logo-cb55bb5c346.png +keywords: +- monitoring +kubeVersion: '>=1.19.0-0' +maintainers: +- email: monotek23@gmail.com + name: monotek +- email: naseem@transit.app + name: naseemkullah +name: alertmanager +sources: +- https://github.com/prometheus/alertmanager +type: application +version: 1.13.0 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/README.md b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/README.md new file mode 100644 index 000000000..d3f4df73a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/README.md @@ -0,0 +1,62 @@ +# Alertmanager + +As per [prometheus.io documentation](https://prometheus.io/docs/alerting/latest/alertmanager/): +> The Alertmanager handles alerts sent by client applications such as the +> Prometheus server. It takes care of deduplicating, grouping, and routing them +> to the correct receiver integration such as email, PagerDuty, or OpsGenie. It +> also takes care of silencing and inhibition of alerts. + +## Prerequisites + +Kubernetes 1.14+ + +## Get Repository Info + +```console +helm repo add prometheus-community https://prometheus-community.github.io/helm-charts +helm repo update +``` + +_See [`helm repo`](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Install Chart + +```console +helm install [RELEASE_NAME] prometheus-community/alertmanager +``` + +_See [configuration](#configuration) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Uninstall Chart + +```console +helm uninstall [RELEASE_NAME] +``` + +This removes all the Kubernetes components associated with the chart and deletes the release. + +_See [helm uninstall](https://helm.sh/docs/helm/helm_uninstall/) for command documentation._ + +## Upgrading Chart + +```console +helm upgrade [RELEASE_NAME] [CHART] --install +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +### To 1.0 + +The [configmap-reload](https://github.com/jimmidyson/configmap-reload) container was replaced by the [prometheus-config-reloader](https://github.com/prometheus-operator/prometheus-operator/tree/main/cmd/prometheus-config-reloader). +Extra command-line arguments specified via configmapReload.prometheus.extraArgs are not compatible and will break with the new prometheus-config-reloader, refer to the [sources](https://github.com/prometheus-operator/prometheus-operator/blob/main/cmd/prometheus-config-reloader/main.go) in order to make the appropriate adjustment to the extea command-line arguments. +The `networking.k8s.io/v1beta1` is no longer supported. use [`networking.k8s.io/v1`](https://kubernetes.io/docs/reference/using-api/deprecation-guide/#ingressclass-v122). + +## Configuration + +See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). To see all configurable options with detailed comments, visit the chart's [values.yaml](./values.yaml), or run these configuration commands: + +```console +helm show values prometheus-community/alertmanager +``` diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml new file mode 100644 index 000000000..cba5de8e2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml @@ -0,0 +1,2 @@ +configmapReload: + enabled: true diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/NOTES.txt new file mode 100644 index 000000000..46ea5bee5 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/NOTES.txt @@ -0,0 +1,21 @@ +1. Get the application URL by running these commands: +{{- if .Values.ingress.enabled }} +{{- range $host := .Values.ingress.hosts }} + {{- range .paths }} + http{{ if $.Values.ingress.tls }}s{{ end }}://{{ $host.host }}{{ .path }} + {{- end }} +{{- end }} +{{- else if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ include "alertmanager.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "alertmanager.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ include "alertmanager.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get --namespace {{ include "alertmanager.namespace" . }} svc -w {{ include "alertmanager.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ include "alertmanager.namespace" . }} {{ include "alertmanager.fullname" . }} --template "{{"{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}"}}") + echo http://$SERVICE_IP:{{ .Values.service.port }} +{{- else if contains "ClusterIP" .Values.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ include "alertmanager.namespace" . }} -l "app.kubernetes.io/name={{ include "alertmanager.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + echo "Visit http://127.0.0.1:{{ .Values.service.port }} to use your application" + kubectl --namespace {{ include "alertmanager.namespace" . }} port-forward $POD_NAME {{ .Values.service.port }}:80 +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/_helpers.tpl new file mode 100644 index 000000000..827b6ee9f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/_helpers.tpl @@ -0,0 +1,92 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "alertmanager.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 "alertmanager.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 "alertmanager.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "alertmanager.labels" -}} +helm.sh/chart: {{ include "alertmanager.chart" . }} +{{ include "alertmanager.selectorLabels" . }} +{{- with .Chart.AppVersion }} +app.kubernetes.io/version: {{ . | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "alertmanager.selectorLabels" -}} +app.kubernetes.io/name: {{ include "alertmanager.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "alertmanager.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "alertmanager.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +Define Ingress apiVersion +*/}} +{{- define "alertmanager.ingress.apiVersion" -}} +{{- printf "networking.k8s.io/v1" }} +{{- end }} + +{{/* +Define Pdb apiVersion +*/}} +{{- define "alertmanager.pdb.apiVersion" -}} +{{- if $.Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} +{{- printf "policy/v1" }} +{{- else }} +{{- printf "policy/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Allow overriding alertmanager namespace +*/}} +{{- define "alertmanager.namespace" -}} +{{- if .Values.namespaceOverride -}} +{{- .Values.namespaceOverride -}} +{{- else -}} +{{- .Release.Namespace -}} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/configmap.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/configmap.yaml new file mode 100644 index 000000000..9e5882dc8 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/configmap.yaml @@ -0,0 +1,21 @@ +{{- if .Values.config.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "alertmanager.fullname" . }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.configAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +data: + alertmanager.yml: | + {{- $config := omit .Values.config "enabled" }} + {{- toYaml $config | default "{}" | nindent 4 }} + {{- range $key, $value := .Values.templates }} + {{ $key }}: |- + {{- $value | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingress.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingress.yaml new file mode 100644 index 000000000..e729a8ad3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingress.yaml @@ -0,0 +1,44 @@ +{{- if .Values.ingress.enabled }} +{{- $fullName := include "alertmanager.fullname" . }} +{{- $svcPort := .Values.service.port }} +apiVersion: {{ include "alertmanager.ingress.apiVersion" . }} +kind: Ingress +metadata: + name: {{ $fullName }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.ingress.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + {{- if .Values.ingress.className }} + ingressClassName: {{ .Values.ingress.className }} + {{- end }} + {{- if .Values.ingress.tls }} + tls: + {{- range .Values.ingress.tls }} + - hosts: + {{- range .hosts }} + - {{ . | quote }} + {{- end }} + secretName: {{ .secretName }} + {{- end }} + {{- end }} + rules: + {{- range .Values.ingress.hosts }} + - host: {{ .host | quote }} + http: + paths: + {{- range .paths }} + - path: {{ .path }} + pathType: {{ .pathType }} + backend: + service: + name: {{ $fullName }} + port: + number: {{ $svcPort }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml new file mode 100644 index 000000000..6f5a02350 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml @@ -0,0 +1,56 @@ +{{- if and .Values.servicePerReplica.enabled .Values.ingressPerReplica.enabled }} +{{- $pathType := .Values.ingressPerReplica.pathType }} +{{- $count := .Values.replicaCount | int -}} +{{- $servicePort := .Values.service.port -}} +{{- $ingressValues := .Values.ingressPerReplica -}} +{{- $fullName := include "alertmanager.fullname" . }} +apiVersion: v1 +kind: List +metadata: + name: {{ $fullName }}-ingressperreplica + namespace: {{ include "alertmanager.namespace" . }} +items: +{{- range $i, $e := until $count }} + - kind: Ingress + apiVersion: {{ include "alertmanager.ingress.apiVersion" $ }} + metadata: + name: {{ $fullName }}-{{ $i }} + namespace: {{ include "alertmanager.namespace" $ }} + labels: + {{- include "alertmanager.labels" $ | nindent 8 }} + {{- if $ingressValues.labels }} +{{ toYaml $ingressValues.labels | indent 8 }} + {{- end }} + {{- if $ingressValues.annotations }} + annotations: +{{ toYaml $ingressValues.annotations | indent 8 }} + {{- end }} + spec: + {{- if $ingressValues.className }} + ingressClassName: {{ $ingressValues.className }} + {{- end }} + rules: + - host: {{ $ingressValues.hostPrefix }}-{{ $i }}.{{ $ingressValues.hostDomain }} + http: + paths: + {{- range $p := $ingressValues.paths }} + - path: {{ tpl $p $ }} + pathType: {{ $pathType }} + backend: + service: + name: {{ $fullName }}-{{ $i }} + port: + name: http + {{- end -}} + {{- if or $ingressValues.tlsSecretName $ingressValues.tlsSecretPerReplica.enabled }} + tls: + - hosts: + - {{ $ingressValues.hostPrefix }}-{{ $i }}.{{ $ingressValues.hostDomain }} + {{- if $ingressValues.tlsSecretPerReplica.enabled }} + secretName: {{ $ingressValues.tlsSecretPerReplica.prefix }}-{{ $i }} + {{- else }} + secretName: {{ $ingressValues.tlsSecretName }} + {{- end }} + {{- end }} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/pdb.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/pdb.yaml new file mode 100644 index 000000000..103e9ecde --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.podDisruptionBudget }} +apiVersion: {{ include "alertmanager.pdb.apiVersion" . }} +kind: PodDisruptionBudget +metadata: + name: {{ include "alertmanager.fullname" . }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + selector: + matchLabels: + {{- include "alertmanager.selectorLabels" . | nindent 6 }} + {{- toYaml .Values.podDisruptionBudget | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml new file mode 100644 index 000000000..bc9ccaaff --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml @@ -0,0 +1,14 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "alertmanager.serviceAccountName" . }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml new file mode 100644 index 000000000..faa75b3ba --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml @@ -0,0 +1,44 @@ +{{- if and .Values.servicePerReplica.enabled }} +{{- $count := .Values.replicaCount | int -}} +{{- $serviceValues := .Values.servicePerReplica -}} +apiVersion: v1 +kind: List +metadata: + name: {{ include "alertmanager.fullname" . }}-serviceperreplica + namespace: {{ include "alertmanager.namespace" . }} +items: +{{- range $i, $e := until $count }} + - apiVersion: v1 + kind: Service + metadata: + name: {{ include "alertmanager.fullname" $ }}-{{ $i }} + namespace: {{ include "alertmanager.namespace" $ }} + labels: + {{- include "alertmanager.labels" $ | nindent 8 }} + {{- if $serviceValues.annotations }} + annotations: +{{ toYaml $serviceValues.annotations | indent 8 }} + {{- end }} + spec: + {{- if $serviceValues.clusterIP }} + clusterIP: {{ $serviceValues.clusterIP }} + {{- end }} + {{- if $serviceValues.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := $serviceValues.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} + {{- end }} + {{- if ne $serviceValues.type "ClusterIP" }} + externalTrafficPolicy: {{ $serviceValues.externalTrafficPolicy }} + {{- end }} + ports: + - name: http + port: {{ $.Values.service.port }} + targetPort: http + selector: + {{- include "alertmanager.selectorLabels" $ | nindent 8 }} + statefulset.kubernetes.io/pod-name: {{ include "alertmanager.fullname" $ }}-{{ $i }} + type: "{{ $serviceValues.type }}" +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/services.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/services.yaml new file mode 100644 index 000000000..eefb9ce16 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/services.yaml @@ -0,0 +1,75 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ include "alertmanager.fullname" . }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.service.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + {{- if .Values.service.ipDualStack.enabled }} + ipFamilies: {{ toYaml .Values.service.ipDualStack.ipFamilies | nindent 4 }} + ipFamilyPolicy: {{ .Values.service.ipDualStack.ipFamilyPolicy }} + {{- end }} + type: {{ .Values.service.type }} + {{- with .Values.service.loadBalancerIP }} + loadBalancerIP: {{ . }} + {{- end }} + {{- with .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := . }} + - {{ $cidr }} + {{- end }} + {{- end }} + ports: + - port: {{ .Values.service.port }} + targetPort: http + protocol: TCP + name: http + {{- if (and (eq .Values.service.type "NodePort") .Values.service.nodePort) }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + {{- with .Values.service.extraPorts }} + {{- toYaml . | nindent 4 }} + {{- end }} + selector: + {{- include "alertmanager.selectorLabels" . | nindent 4 }} +--- +apiVersion: v1 +kind: Service +metadata: + name: {{ include "alertmanager.fullname" . }}-headless + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + clusterIP: None + ports: + - port: {{ .Values.service.port }} + targetPort: http + protocol: TCP + name: http + {{- if or (gt (int .Values.replicaCount) 1) (.Values.additionalPeers) }} + - port: {{ .Values.service.clusterPort }} + targetPort: clusterpeer-tcp + protocol: TCP + name: cluster-tcp + - port: {{ .Values.service.clusterPort }} + targetPort: clusterpeer-udp + protocol: UDP + name: cluster-udp + {{- end }} + {{- with .Values.service.extraPorts }} + {{- toYaml . | nindent 4 }} + {{- end }} + selector: + {{- include "alertmanager.selectorLabels" . | nindent 4 }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/statefulset.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/statefulset.yaml new file mode 100644 index 000000000..0bc765c2f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/statefulset.yaml @@ -0,0 +1,254 @@ +{{- $svcClusterPort := .Values.service.clusterPort }} +apiVersion: apps/v1 +kind: StatefulSet +metadata: + name: {{ include "alertmanager.fullname" . }} + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.statefulSet.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + replicas: {{ .Values.replicaCount }} + minReadySeconds: {{ .Values.minReadySeconds }} + revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} + selector: + matchLabels: + {{- include "alertmanager.selectorLabels" . | nindent 6 }} + serviceName: {{ include "alertmanager.fullname" . }}-headless + template: + metadata: + labels: + {{- include "alertmanager.selectorLabels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + annotations: + {{- if not .Values.configmapReload.enabled }} + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "alertmanager.serviceAccountName" . }} + {{- with .Values.dnsConfig }} + dnsConfig: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.hostAliases }} + hostAliases: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.schedulerName }} + schedulerName: {{ . }} + {{- end }} + {{- if or .Values.podAntiAffinity .Values.affinity }} + affinity: + {{- end }} + {{- with .Values.affinity }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if eq .Values.podAntiAffinity "hard" }} + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: {{ .Values.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ include "alertmanager.name" . }}]} + {{- else if eq .Values.podAntiAffinity "soft" }} + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + topologyKey: {{ .Values.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ include "alertmanager.name" . }}]} + {{- end }} + {{- with .Values.priorityClassName }} + priorityClassName: {{ . }} + {{- end }} + {{- with .Values.topologySpreadConstraints }} + topologySpreadConstraints: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + securityContext: + {{- toYaml .Values.podSecurityContext | nindent 8 }} + {{- with .Values.extraInitContainers }} + initContainers: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + {{- if .Values.configmapReload.enabled }} + - name: {{ .Chart.Name }}-{{ .Values.configmapReload.name }} + image: "{{ .Values.configmapReload.image.repository }}:{{ .Values.configmapReload.image.tag }}" + imagePullPolicy: "{{ .Values.configmapReload.image.pullPolicy }}" + {{- with .Values.configmapReload.extraEnv }} + env: + {{- toYaml . | nindent 12 }} + {{- end }} + args: + {{- if and (hasKey .Values.configmapReload.extraArgs "config-file" | not) (hasKey .Values.configmapReload.extraArgs "watched-dir" | not) }} + - --watched-dir=/etc/alertmanager + {{- end }} + {{- if not (hasKey .Values.configmapReload.extraArgs "reload-url") }} + - --reload-url=http://127.0.0.1:9093/-/reload + {{- end }} + {{- range $key, $value := .Values.configmapReload.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + resources: + {{- toYaml .Values.configmapReload.resources | nindent 12 }} + {{- with .Values.configmapReload.containerPort }} + ports: + - containerPort: {{ . }} + {{- end }} + {{- with .Values.configmapReload.securityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config + mountPath: /etc/alertmanager + {{- if .Values.configmapReload.extraVolumeMounts }} + {{- toYaml .Values.configmapReload.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- end }} + - name: {{ .Chart.Name }} + securityContext: + {{- toYaml .Values.securityContext | nindent 12 }} + image: "{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + env: + - name: POD_IP + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: status.podIP + {{- if .Values.extraEnv }} + {{- toYaml .Values.extraEnv | nindent 12 }} + {{- end }} + {{- with .Values.command }} + command: + {{- toYaml . | nindent 12 }} + {{- end }} + args: + - --storage.path=/alertmanager + {{- if not (hasKey .Values.extraArgs "config.file") }} + - --config.file=/etc/alertmanager/alertmanager.yml + {{- end }} + {{- if or (gt (int .Values.replicaCount) 1) (.Values.additionalPeers) }} + - --cluster.advertise-address=[$(POD_IP)]:{{ $svcClusterPort }} + - --cluster.listen-address=0.0.0.0:{{ $svcClusterPort }} + {{- end }} + {{- if gt (int .Values.replicaCount) 1}} + {{- $fullName := include "alertmanager.fullname" . }} + {{- range $i := until (int .Values.replicaCount) }} + - --cluster.peer={{ $fullName }}-{{ $i }}.{{ $fullName }}-headless:{{ $svcClusterPort }} + {{- end }} + {{- end }} + {{- if .Values.additionalPeers }} + {{- range $item := .Values.additionalPeers }} + - --cluster.peer={{ $item }} + {{- end }} + {{- end }} + {{- range $key, $value := .Values.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.baseURL }} + - --web.external-url={{ .Values.baseURL }} + {{- end }} + ports: + - name: http + containerPort: 9093 + protocol: TCP + {{- if or (gt (int .Values.replicaCount) 1) (.Values.additionalPeers) }} + - name: clusterpeer-tcp + containerPort: {{ $svcClusterPort }} + protocol: TCP + - name: clusterpeer-udp + containerPort: {{ $svcClusterPort }} + protocol: UDP + {{- end }} + livenessProbe: + {{- toYaml .Values.livenessProbe | nindent 12 }} + readinessProbe: + {{- toYaml .Values.readinessProbe | nindent 12 }} + resources: + {{- toYaml .Values.resources | nindent 12 }} + volumeMounts: + {{- if .Values.config.enabled }} + - name: config + mountPath: /etc/alertmanager + {{- end }} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + - name: storage + mountPath: /alertmanager + {{- if .Values.extraVolumeMounts }} + {{- toYaml .Values.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- with .Values.extraContainers }} + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + {{- if .Values.config.enabled }} + - name: config + configMap: + name: {{ include "alertmanager.fullname" . }} + {{- end }} + {{- range .Values.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- if .Values.extraVolumes }} + {{- toYaml .Values.extraVolumes | nindent 8 }} + {{- end }} + {{- if .Values.extraPodConfigs }} + {{- toYaml .Values.extraPodConfigs | nindent 6 }} + {{- end }} + {{- if .Values.persistence.enabled }} + volumeClaimTemplates: + - metadata: + name: storage + spec: + accessModes: + {{- toYaml .Values.persistence.accessModes | nindent 10 }} + resources: + requests: + storage: {{ .Values.persistence.size }} + {{- if .Values.persistence.storageClass }} + {{- if (eq "-" .Values.persistence.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: {{ .Values.persistence.storageClass }} + {{- end }} + {{- end }} + {{- else }} + - name: storage + emptyDir: {} + {{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml new file mode 100644 index 000000000..410eba5bd --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml @@ -0,0 +1,20 @@ +{{- if .Values.testFramework.enabled }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "alertmanager.fullname" . }}-test-connection" + labels: + {{- include "alertmanager.labels" . | nindent 4 }} + {{- with .Values.testFramework.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ include "alertmanager.namespace" . }} +spec: + containers: + - name: wget + image: busybox + command: ['wget'] + args: ['{{ include "alertmanager.fullname" . }}:{{ .Values.service.port }}'] + restartPolicy: Never +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/vpa.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/vpa.yaml new file mode 100644 index 000000000..53f70a28e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/templates/vpa.yaml @@ -0,0 +1,26 @@ +{{- if .Values.verticalPodAutoscaler.enabled }} +apiVersion: autoscaling.k8s.io/v1 +kind: VerticalPodAutoscaler +metadata: + name: {{ include "alertmanager.fullname" . }} + namespace: {{ .Release.Namespace }} +spec: + {{- if .Values.verticalPodAutoscaler.recommenders }} + recommenders: + {{- range .Values.verticalPodAutoscaler.recommenders }} + - name: {{ .name }} + {{- end }} + {{- end }} + targetRef: + apiVersion: apps/v1 + kind: StatefulSet + name: {{ include "alertmanager.fullname" . }} + {{- if .Values.verticalPodAutoscaler.updatePolicy }} + updatePolicy: + {{- toYaml .Values.verticalPodAutoscaler.updatePolicy | nindent 4 }} + {{- end }} + {{- if .Values.verticalPodAutoscaler.resourcePolicy }} + resourcePolicy: + {{- toYaml .Values.verticalPodAutoscaler.resourcePolicy | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.schema.json b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.schema.json new file mode 100644 index 000000000..d3be015cf --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.schema.json @@ -0,0 +1,946 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema", + "title": "alertmanager", + "description": "The Alertmanager handles alerts sent by client applications such as the Prometheus server.", + "type": "object", + "required": [ + "replicaCount", + "image", + "serviceAccount", + "service", + "persistence", + "config" + ], + "definitions": { + "image": { + "description": "Container image parameters.", + "type": "object", + "required": ["repository"], + "additionalProperties": false, + "properties": { + "repository": { + "description": "Image repository. Path to the image with registry(quay.io) or without(prometheus/alertmanager) for docker.io.", + "type": "string" + }, + "pullPolicy": { + "description": "Image pull policy. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated.", + "type": "string", + "enum": [ + "Never", + "IfNotPresent", + "Always" + ], + "default": "IfNotPresent" + }, + "tag": { + "description": "Use chart appVersion by default.", + "type": "string", + "default": "" + } + } + }, + "resources": { + "description": "Resource limits and requests for the Container.", + "type": "object", + "properties": { + "limits": { + "description": "Resource limits for the Container.", + "type": "object", + "properties": { + "cpu": { + "description": "CPU request for the Container.", + "type": "string" + }, + "memory": { + "description": "Memory request for the Container.", + "type": "string" + } + } + }, + "requests": { + "description": "Resource requests for the Container.", + "type": "object", + "properties": { + "cpu": { + "description": "CPU request for the Container.", + "type": "string" + }, + "memory": { + "description": "Memory request for the Container.", + "type": "string" + } + } + } + } + }, + "securityContext": { + "description": "Security context for the container.", + "type": "object", + "properties": { + "capabilities": { + "description": "Specifies the capabilities to be dropped by the container.", + "type": "object", + "properties": { + "drop": { + "description": "List of capabilities to be dropped.", + "type": "array", + "items": { + "type": "string" + } + } + } + }, + "readOnlyRootFilesystem": { + "description": "Specifies whether the root file system should be mounted as read-only.", + "type": "boolean" + }, + "runAsUser": { + "description": "Specifies the UID (User ID) to run the container as.", + "type": "integer" + }, + "runAsNonRoot": { + "description": "Specifies whether to run the container as a non-root user.", + "type": "boolean" + }, + "runAsGroup": { + "description": "Specifies the GID (Group ID) to run the container as.", + "type": "integer" + } + } + }, + "volumeMounts": { + "description": "List of volume mounts for the Container.", + "type": "array", + "items": { + "description": "Volume mounts for the Container.", + "type": "object", + "required": ["name", "mountPath"], + "properties": { + "name": { + "description": "The name of the volume to mount.", + "type": "string" + }, + "mountPath": { + "description": "The mount path for the volume.", + "type": "string" + }, + "readOnly": { + "description": "Specifies if the volume should be mounted in read-only mode.", + "type": "boolean" + } + } + } + }, + "env": { + "description": "List of environment variables for the Container.", + "type": "array", + "items": { + "description": "Environment variables for the Container.", + "type": "object", + "required": ["name"], + "properties": { + "name": { + "description": "The name of the environment variable.", + "type": "string" + }, + "value": { + "description": "The value of the environment variable.", + "type": "string" + } + } + } + }, + "config": { + "description": "https://prometheus.io/docs/alerting/latest/configuration/", + "duration": { + "type": "string", + "pattern": "^((([0-9]+)y)?(([0-9]+)w)?(([0-9]+)d)?(([0-9]+)h)?(([0-9]+)m)?(([0-9]+)s)?(([0-9]+)ms)?|0)$" + }, + "labelname": { + "type": "string", + "pattern": "^[a-zA-Z_][a-zA-Z0-9_]*$|^...$" + }, + "route": { + "description": "Alert routing configuration.", + "type": "object", + "properties": { + "receiver": { + "description": "The default receiver to send alerts to.", + "type": "string" + }, + "group_by": { + "description": "The labels by which incoming alerts are grouped together.", + "type": "array", + "items": { + "type": "string", + "$ref": "#/definitions/config/labelname" + } + }, + "continue": { + "description": "Whether an alert should continue matching subsequent sibling nodes.", + "type": "boolean", + "default": false + }, + "matchers": { + "description": "A list of matchers that an alert has to fulfill to match the node.", + "type": "array", + "items": { + "type": "string" + } + }, + "group_wait": { + "description": "How long to initially wait to send a notification for a group of alerts.", + "$ref": "#/definitions/config/duration" + }, + "group_interval": { + "description": "How long to wait before sending a notification about new alerts that are added to a group of alerts for which an initial notification has already been sent.", + "$ref": "#/definitions/config/duration" + }, + "repeat_interval": { + "description": "How long to wait before sending a notification again if it has already been sent successfully for an alert.", + "$ref": "#/definitions/config/duration" + }, + "mute_time_intervals": { + "description": "Times when the route should be muted.", + "type": "array", + "items": { + "type": "string" + } + }, + "active_time_intervals": { + "description": "Times when the route should be active.", + "type": "array", + "items": { + "type": "string" + } + }, + "routes": { + "description": "Zero or more child routes.", + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/config/route" + } + } + } + } + } + }, + "properties": { + "replicaCount": { + "description": "Number of desired pods.", + "type": "integer", + "default": 1, + "minimum": 0 + }, + "image": { + "description": "Container image parameters.", + "$ref": "#/definitions/image" + }, + "baseURL": { + "description": "External URL where alertmanager is reachable.", + "type": "string", + "default": "", + "examples": [ + "https://alertmanager.example.com" + ] + }, + "extraArgs": { + "description": "Additional alertmanager container arguments. Use args without '--', only 'key: value' syntax.", + "type": "object", + "default": {} + }, + "extraSecretMounts": { + "description": "Additional Alertmanager Secret mounts.", + "type": "array", + "default": [], + "items": { + "type": "object", + "required": ["name", "mountPath", "secretName"], + "properties": { + "name": { + "type": "string" + }, + "mountPath": { + "type": "string" + }, + "subPath": { + "type": "string", + "default": "" + }, + "secretName": { + "type": "string" + }, + "readOnly": { + "type": "boolean", + "default": false + } + } + } + }, + "imagePullSecrets": { + "description": "The property allows you to configure multiple image pull secrets.", + "type": "array", + "default": [], + "items": { + "type": "object", + "required": ["name"], + "properties": { + "name": { + "description": "Specifies the Secret name of the image pull secret.", + "type": "string" + } + } + } + }, + "nameOverride": { + "description": "Override value for the name of the Helm chart.", + "type": "string", + "default": "" + }, + "fullnameOverride": { + "description": "Override value for the fully qualified app name.", + "type": "string", + "default": "" + }, + "namespaceOverride": { + "description": "Override deployment namespace.", + "type": "string", + "default": "" + }, + "automountServiceAccountToken": { + "description": "Specifies whether to automatically mount the ServiceAccount token into the Pod's filesystem.", + "type": "boolean", + "default": true + }, + "serviceAccount": { + "description": "Contains properties related to the service account configuration.", + "type": "object", + "required": ["create"], + "properties": { + "create": { + "description": "Specifies whether a service account should be created.", + "type": "boolean", + "default": true + }, + "annotations": { + "description": "Annotations to add to the service account.", + "type": "object", + "default": {} + }, + "name": { + "description": "The name of the service account to use. If not set and create is true, a name is generated using the fullname template.", + "type": "string", + "default": "" + } + } + }, + "schedulerName": { + "description": "Sets the schedulerName in the alertmanager pod.", + "type": "string", + "default": "" + }, + "priorityClassName": { + "description": "Sets the priorityClassName in the alertmanager pod.", + "type": "string", + "default": "" + }, + "podSecurityContext": { + "description": "Pod security context configuration.", + "type": "object", + "properties": { + "fsGroup": { + "description": "The fsGroup value for the pod's security context.", + "type": "integer", + "default": 65534 + }, + "runAsUser": { + "description": "The UID to run the pod's containers as.", + "type": "integer" + }, + "runAsGroup": { + "description": "The GID to run the pod's containers as.", + "type": "integer" + } + } + }, + "dnsConfig": { + "description": "DNS configuration for the pod.", + "type": "object", + "properties": { + "nameservers": { + "description": "List of DNS server IP addresses.", + "type": "array", + "items": { + "type": "string" + } + }, + "searches": { + "description": "List of DNS search domains.", + "type": "array", + "items": { + "type": "string" + } + }, + "options": { + "description": "List of DNS options.", + "type": "array", + "items": { + "description": "DNS options.", + "type": "object", + "required": ["name"], + "properties": { + "name": { + "description": "The name of the DNS option.", + "type": "string" + }, + "value": { + "description": "The value of the DNS option.", + "type": "string" + } + } + } + } + } + }, + "hostAliases": { + "description": "List of host aliases.", + "type": "array", + "items": { + "description": "Host aliases configuration.", + "type": "object", + "required": ["ip", "hostnames"], + "properties": { + "ip": { + "description": "IP address associated with the host alias.", + "type": "string" + }, + "hostnames": { + "description": "List of hostnames associated with the IP address.", + "type": "array", + "items": { + "type": "string" + } + } + } + } + }, + "securityContext": { + "description": "Security context for the container.", + "$ref": "#/definitions/securityContext" + }, + "additionalPeers": { + "description": "Additional peers for a alertmanager.", + "type": "array", + "items": { + "type": "string" + } + }, + "extraInitContainers": { + "description": "Additional InitContainers to initialize the pod.", + "type": "array", + "default": [], + "items": { + "required": ["name", "image"], + "properties": { + "name": { + "description": "The name of the InitContainer.", + "type": "string" + }, + "image": { + "description": "The container image to use for the InitContainer.", + "type": "string" + }, + "pullPolicy": { + "description": "Image pull policy. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated.", + "type": "string", + "enum": [ + "Never", + "IfNotPresent", + "Always" + ], + "default": "IfNotPresent" + }, + "command": { + "description": "The command to run in the InitContainer.", + "type": "array", + "items": { + "type": "string" + } + }, + "args": { + "description": "Additional command arguments for the InitContainer.", + "type": "array", + "items": { + "type": "string" + } + }, + "ports": { + "description": "List of ports to expose from the container.", + "type": "array", + "items": { + "type": "object" + } + }, + "env": { + "description": "List of environment variables for the InitContainer.", + "$ref": "#/definitions/env" + }, + "envFrom": { + "description": "List of sources to populate environment variables in the container.", + "type": "array", + "items": { + "type": "object" + } + }, + "volumeMounts": { + "description": "List of volume mounts for the InitContainer.", + "$ref": "#/definitions/volumeMounts" + }, + "resources": { + "description": "Resource requirements for the InitContainer.", + "$ref": "#/definitions/resources" + }, + "securityContext": { + "$ref": "#/definitions/securityContext", + "description": "The security context for the InitContainer." + } + } + } + }, + "extraContainers": { + "description": "Additional containers to add to the stateful set.", + "type": "array", + "default": [], + "items": { + "required": ["name", "image"], + "properties": { + "name": { + "description": "The name of the InitContainer.", + "type": "string" + }, + "image": { + "description": "The container image to use for the InitContainer.", + "type": "string" + }, + "pullPolicy": { + "description": "Image pull policy. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated.", + "type": "string", + "enum": [ + "Never", + "IfNotPresent", + "Always" + ], + "default": "IfNotPresent" + }, + "command": { + "description": "The command to run in the InitContainer.", + "type": "array", + "items": { + "type": "string" + } + }, + "args": { + "description": "Additional command arguments for the InitContainer.", + "type": "array", + "items": { + "type": "string" + } + }, + "ports": { + "description": "List of ports to expose from the container.", + "type": "array", + "items": { + "type": "object" + } + }, + "env": { + "description": "List of environment variables for the InitContainer.", + "$ref": "#/definitions/env" + }, + "envFrom": { + "description": "List of sources to populate environment variables in the container.", + "type": "array", + "items": { + "type": "object" + } + }, + "volumeMounts": { + "description": "List of volume mounts for the InitContainer.", + "$ref": "#/definitions/volumeMounts" + }, + "resources": { + "description": "Resource requirements for the InitContainer.", + "$ref": "#/definitions/resources" + }, + "securityContext": { + "$ref": "#/definitions/securityContext", + "description": "The security context for the InitContainer." + } + } + } + }, + "resources": { + "description": "Resource limits and requests for the pod.", + "$ref": "#/definitions/resources" + }, + "livenessProbe": { + "description": "Liveness probe configuration.", + "type": "object" + }, + "readinessProbe": { + "description": "Readiness probe configuration.", + "type": "object" + }, + "service": { + "description": "Service configuration.", + "type": "object", + "required": ["type", "port"], + "properties": { + "annotations": { + "description": "Annotations to add to the service.", + "type": "object" + }, + "type": { + "description": "Service type.", + "type": "string" + }, + "port": { + "description": "Port number for the service.", + "type": "integer" + }, + "clusterPort": { + "description": "Port number for the cluster.", + "type": "integer" + }, + "loadBalancerIP": { + "description": "External IP to assign when the service type is LoadBalancer.", + "type": "string" + }, + "loadBalancerSourceRanges": { + "description": "IP ranges to allow access to the loadBalancerIP.", + "type": "array", + "items": { + "type": "string" + } + }, + "nodePort": { + "description": "Specific nodePort to force when service type is NodePort.", + "type": "integer" + } + } + }, + "ingress": { + "description": "Ingress configuration.", + "type": "object", + "properties": { + "enabled": { + "description": "Indicates if Ingress is enabled.", + "type": "boolean" + }, + "className": { + "description": "Ingress class name.", + "type": "string" + }, + "annotations": { + "description": "Annotations to add to the Ingress.", + "type": "object" + }, + "hosts": { + "description": "Host and path configuration for the Ingress.", + "type": "array", + "items": { + "type": "object", + "properties": { + "host": { + "description": "Host name for the Ingress.", + "type": "string" + }, + "paths": { + "description": "Path configuration for the Ingress.", + "type": "array", + "items": { + "type": "object", + "properties": { + "path": { + "description": "Path for the Ingress.", + "type": "string" + }, + "pathType": { + "description": "Path type for the Ingress.", + "type": "string" + } + } + } + } + } + } + }, + "tls": { + "description": "TLS configuration for the Ingress.", + "type": "array", + "items": { + "type": "object", + "properties": { + "secretName": { + "description": "Name of the secret for TLS.", + "type": "string" + }, + "hosts": { + "description": "Host names for the TLS configuration.", + "type": "array", + "items": { + "type": "string" + } + } + } + } + } + } + }, + "nodeSelector": { + "description": "Node selector for pod assignment.", + "type": "object" + }, + "tolerations": { + "description": "Tolerations for pod assignment.", + "type": "array" + }, + "affinity": { + "description": "Affinity rules for pod assignment.", + "type": "object" + }, + "podAntiAffinity": { + "description": "Pod anti-affinity configuration.", + "type": "string", + "enum": ["", "soft", "hard"], + "default": "" + }, + "podAntiAffinityTopologyKey": { + "description": "Topology key to use for pod anti-affinity.", + "type": "string" + }, + "topologySpreadConstraints": { + "description": "Topology spread constraints for pod assignment.", + "type": "array", + "items": { + "type": "object", + "required": ["maxSkew", "topologyKey", "whenUnsatisfiable", "labelSelector"], + "properties": { + "maxSkew": { + "type": "integer" + }, + "topologyKey": { + "type": "string" + }, + "whenUnsatisfiable": { + "type": "string", + "enum": ["DoNotSchedule", "ScheduleAnyway"] + }, + "labelSelector": { + "type": "object", + "required": ["matchLabels"], + "properties": { + "matchLabels": { + "type": "object" + } + } + } + } + } + }, + "statefulSet": { + "description": "StatefulSet configuration for managing pods.", + "type": "object", + "properties": { + "annotations": { + "type": "object" + } + } + }, + "podAnnotations": { + "description": "Annotations to add to the pods.", + "type": "object" + }, + "podLabels": { + "description": "Labels to add to the pods.", + "type": "object" + }, + "podDisruptionBudget": { + "description": "Pod disruption budget configuration.", + "type": "object", + "properties": { + "maxUnavailable": { + "type": "integer" + }, + "minAvailable": { + "type": "integer" + } + } + }, + "command": { + "description": "The command to be executed in the container.", + "type": "array", + "items": { + "type": "string" + } + }, + "persistence": { + "description": "Persistence configuration for storing data.", + "type": "object", + "required": ["enabled", "size"], + "properties": { + "enabled": { + "type": "boolean" + }, + "storageClass": { + "type": "string" + }, + "accessModes": { + "type": "array", + "items": { + "type": "string" + } + }, + "size": { + "type": "string" + } + } + }, + "configAnnotations": { + "description": "Annotations to be added to the Alertmanager configuration.", + "type": "object" + }, + "config": { + "description": "Alertmanager configuration.", + "type": "object", + "properties": { + "enabled": { + "description": "Whether to create alermanager configmap or not.", + "type": "boolean" + }, + "global": { + "description": "Global configuration options.", + "type": "object" + }, + "templates": { + "description": "Alertmanager template files.", + "type": "array", + "items": { + "type": "string" + } + }, + "receivers": { + "description": "Alert receivers configuration.", + "type": "array", + "items": { + "type": "object", + "required": ["name"], + "properties": { + "name": { + "description": "The unique name of the receiver.", + "type": "string" + } + } + } + }, + "route": { + "description": "Alert routing configuration.", + "type": "object", + "$ref": "#/definitions/config/route" + } + } + }, + "configmapReload": { + "description": "Monitors ConfigMap changes and POSTs to a URL.", + "type": "object", + "properties": { + "enabled": { + "description": "Specifies whether the configmap-reload container should be deployed.", + "type": "boolean", + "default": false + }, + "name": { + "description": "The name of the configmap-reload container.", + "type": "string" + }, + "image": { + "description": "The container image for the configmap-reload container.", + "$ref": "#/definitions/image" + }, + "containerPort": { + "description": "Port number for the configmap-reload container.", + "type": "integer" + }, + "resources": { + "description": "Resource requests and limits for the configmap-reload container.", + "$ref": "#/definitions/resources" + } + } + }, + "templates": { + "description": "Custom templates used by Alertmanager.", + "type": "object" + }, + "extraVolumeMounts": { + "description": "List of volume mounts for the Container.", + "$ref": "#/definitions/volumeMounts" + }, + "extraVolumes": { + "description": "Additional volumes to be mounted in the Alertmanager pod.", + "type": "array", + "default": [], + "items": { + "type": "object", + "required": ["name"], + "properties": { + "name": { + "type": "string" + } + } + } + }, + "extraEnv": { + "description": "List of environment variables for the Container.", + "$ref": "#/definitions/env" + }, + "testFramework": { + "description": "Configuration for the test Pod.", + "type": "object", + "properties": { + "enabled": { + "description": "Specifies whether the test Pod is enabled.", + "type": "boolean", + "default": false + }, + "annotations": { + "description": "Annotations to be added to the test Pod.", + "type": "object" + } + } + }, + "verticalPodAutoscaler": { + "description": "Vertical Pod Autoscaling configuration.", + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "default": false + }, + "recommenders": { + "type": "array" + }, + "updatePolicy": { + "type": "object" + }, + "resourcePolicy": { + "type": "object" + } + } + }, + "extraPodConfigs": { + "description": "Object to allow users to add additional Pod configuration like dnsPolicy or hostNetwork", + "type": "object" + } + } +} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.yaml new file mode 100644 index 000000000..133f438c9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/alertmanager/values.yaml @@ -0,0 +1,404 @@ +# yaml-language-server: $schema=values.schema.json +# Default values for alertmanager. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +replicaCount: 1 + +# Number of old history to retain to allow rollback +# Default Kubernetes value is set to 10 +revisionHistoryLimit: 10 + +image: + repository: quay.io/prometheus/alertmanager + pullPolicy: IfNotPresent + # Overrides the image tag whose default is the chart appVersion. + tag: "" + +# Full external URL where alertmanager is reachable, used for backlinks. +baseURL: "" + +extraArgs: {} + +## Additional Alertmanager Secret mounts +# Defines additional mounts with secrets. Secrets must be manually created in the namespace. +extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # subPath: "" + # secretName: alertmanager-secret-files + # readOnly: true + +imagePullSecrets: [] +nameOverride: "" +fullnameOverride: "" +## namespaceOverride overrides the namespace which the resources will be deployed in +namespaceOverride: "" + +automountServiceAccountToken: true + +serviceAccount: + # Specifies whether a service account should be created + create: true + # Annotations to add to the service account + annotations: {} + # The name of the service account to use. + # If not set and create is true, a name is generated using the fullname template + name: "" + +# Sets priorityClassName in alertmanager pod +priorityClassName: "" + +# Sets schedulerName in alertmanager pod +schedulerName: "" + +podSecurityContext: + fsGroup: 65534 +dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 +hostAliases: [] + # - ip: "127.0.0.1" + # hostnames: + # - "foo.local" + # - "bar.local" + # - ip: "10.1.2.3" + # hostnames: + # - "foo.remote" + # - "bar.remote" +securityContext: + # capabilities: + # drop: + # - ALL + # readOnlyRootFilesystem: true + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + +additionalPeers: [] + +## Additional InitContainers to initialize the pod +## +extraInitContainers: [] + +## Additional containers to add to the stateful set. This will allow to setup sidecarContainers like a proxy to integrate +## alertmanager with an external tool like teams that has not direct integration. +## +extraContainers: [] + +livenessProbe: + httpGet: + path: / + port: http + +readinessProbe: + httpGet: + path: / + port: http + +service: + annotations: {} + labels: {} + type: ClusterIP + port: 9093 + clusterPort: 9094 + loadBalancerIP: "" # Assign ext IP when Service type is LoadBalancer + loadBalancerSourceRanges: [] # Only allow access to loadBalancerIP from these IPs + # if you want to force a specific nodePort. Must be use with service.type=NodePort + # nodePort: + + # Optionally specify extra list of additional ports exposed on both services + extraPorts: [] + + # ip dual stack + ipDualStack: + enabled: false + ipFamilies: ["IPv6", "IPv4"] + ipFamilyPolicy: "PreferDualStack" + +# Configuration for creating a separate Service for each statefulset Alertmanager replica +# +servicePerReplica: + enabled: false + annotations: {} + + # Loadbalancer source IP ranges + # Only used if servicePerReplica.type is "LoadBalancer" + loadBalancerSourceRanges: [] + + # Denotes if this Service desires to route external traffic to node-local or cluster-wide endpoints + # + externalTrafficPolicy: Cluster + + # Service type + # + type: ClusterIP + +ingress: + enabled: false + className: "" + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + hosts: + - host: alertmanager.domain.com + paths: + - path: / + pathType: ImplementationSpecific + tls: [] + # - secretName: chart-example-tls + # hosts: + # - alertmanager.domain.com + +# Configuration for creating an Ingress that will map to each Alertmanager replica service +# alertmanager.servicePerReplica must be enabled +# +ingressPerReplica: + enabled: false + + # className for the ingresses + # + className: "" + + annotations: {} + labels: {} + + # Final form of the hostname for each per replica ingress is + # {{ ingressPerReplica.hostPrefix }}-{{ $replicaNumber }}.{{ ingressPerReplica.hostDomain }} + # + # Prefix for the per replica ingress that will have `-$replicaNumber` + # appended to the end + hostPrefix: "alertmanager" + # Domain that will be used for the per replica ingress + hostDomain: "domain.com" + + # Paths to use for ingress rules + # + paths: + - / + + # PathType for ingress rules + # + pathType: ImplementationSpecific + + # Secret name containing the TLS certificate for alertmanager per replica ingress + # Secret must be manually created in the namespace + tlsSecretName: "" + + # Separated secret for each per replica Ingress. Can be used together with cert-manager + # + tlsSecretPerReplica: + enabled: false + # Final form of the secret for each per replica ingress is + # {{ tlsSecretPerReplica.prefix }}-{{ $replicaNumber }} + # + prefix: "alertmanager" + +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: 10m + # memory: 32Mi + +nodeSelector: {} + +tolerations: [] + +affinity: {} + +## Pod anti-affinity can prevent the scheduler from placing Alertmanager replicas on the same node. +## The default value "soft" means that the scheduler should *prefer* to not schedule two replica pods onto the same node but no guarantee is provided. +## The value "hard" means that the scheduler is *required* to not schedule two replica pods onto the same node. +## The value "" will disable pod anti-affinity so that no anti-affinity rules will be configured. +## +podAntiAffinity: "" + +## If anti-affinity is enabled sets the topologyKey to use for anti-affinity. +## This can be changed to, for example, failure-domain.beta.kubernetes.io/zone +## +podAntiAffinityTopologyKey: kubernetes.io/hostname + +## Topology spread constraints rely on node labels to identify the topology domain(s) that each Node is in. +## Ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ +topologySpreadConstraints: [] + # - maxSkew: 1 + # topologyKey: failure-domain.beta.kubernetes.io/zone + # whenUnsatisfiable: DoNotSchedule + # labelSelector: + # matchLabels: + # app.kubernetes.io/instance: alertmanager + +statefulSet: + annotations: {} + +## Minimum number of seconds for which a newly created pod should be ready without any of its container crashing for it to +## be considered available. Defaults to 0 (pod will be considered available as soon as it is ready). +## This is an alpha field from kubernetes 1.22 until 1.24 which requires enabling the StatefulSetMinReadySeconds +## feature gate. +## Ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#minimum-ready-seconds +minReadySeconds: 0 + +podAnnotations: {} +podLabels: {} + +# Ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ +podDisruptionBudget: {} + # maxUnavailable: 1 + # minAvailable: 1 + +command: [] + +persistence: + enabled: true + ## Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. + ## + # storageClass: "-" + accessModes: + - ReadWriteOnce + size: 50Mi + +configAnnotations: {} + ## For example if you want to provide private data from a secret vault + ## https://github.com/banzaicloud/bank-vaults/tree/main/charts/vault-secrets-webhook + ## P.s.: Add option `configMapMutation: true` for vault-secrets-webhook + # vault.security.banzaicloud.io/vault-role: "admin" + # vault.security.banzaicloud.io/vault-addr: "https://vault.vault.svc.cluster.local:8200" + # vault.security.banzaicloud.io/vault-skip-verify: "true" + # vault.security.banzaicloud.io/vault-path: "kubernetes" + ## Example for inject secret + # slack_api_url: '${vault:secret/data/slack-hook-alerts#URL}' + +config: + enabled: true + global: {} + # slack_api_url: '' + + templates: + - '/etc/alertmanager/*.tmpl' + + receivers: + - name: default-receiver + # slack_configs: + # - channel: '@you' + # send_resolved: true + + route: + group_wait: 10s + group_interval: 5m + receiver: default-receiver + repeat_interval: 3h + +## Monitors ConfigMap changes and POSTs to a URL +## Ref: https://github.com/prometheus-operator/prometheus-operator/tree/main/cmd/prometheus-config-reloader +## +configmapReload: + ## If false, the configmap-reload container will not be deployed + ## + enabled: false + + ## configmap-reload container name + ## + name: configmap-reload + + ## configmap-reload container image + ## + image: + repository: quay.io/prometheus-operator/prometheus-config-reloader + tag: v0.66.0 + pullPolicy: IfNotPresent + + # containerPort: 9533 + + ## configmap-reload resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: {} + + extraArgs: {} + + ## Optionally specify extra list of additional volumeMounts + extraVolumeMounts: [] + # - name: extras + # mountPath: /usr/share/extras + # readOnly: true + + ## Optionally specify extra environment variables to add to alertmanager container + extraEnv: [] + # - name: FOO + # value: BAR + + securityContext: {} + # capabilities: + # drop: + # - ALL + # readOnlyRootFilesystem: true + # runAsUser: 65534 + # runAsNonRoot: true + # runAsGroup: 65534 + +templates: {} +# alertmanager.tmpl: |- + +## Optionally specify extra list of additional volumeMounts +extraVolumeMounts: [] + # - name: extras + # mountPath: /usr/share/extras + # readOnly: true + +## Optionally specify extra list of additional volumes +extraVolumes: [] + # - name: extras + # emptyDir: {} + +## Optionally specify extra environment variables to add to alertmanager container +extraEnv: [] + # - name: FOO + # value: BAR + +testFramework: + enabled: false + annotations: + "helm.sh/hook": test-success + # "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + +# --- Vertical Pod Autoscaler +verticalPodAutoscaler: + # -- Use VPA for alertmanager + enabled: false + # recommenders: + # - name: 'alternative' + # updatePolicy: + # updateMode: "Auto" + # minReplicas: 1 + # resourcePolicy: + # containerPolicies: + # - containerName: '*' + # minAllowed: + # cpu: 100m + # memory: 128Mi + # maxAllowed: + # cpu: 1 + # memory: 500Mi + # controlledResources: ["cpu", "memory"] + +# --- Extra Pod Configs +extraPodConfigs: {} + # dnsPolicy: ClusterFirstWithHostNet + # hostNetwork: true diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/.helmignore b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/.helmignore new file mode 100644 index 000000000..f0c131944 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/.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/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/Chart.yaml new file mode 100644 index 000000000..755213319 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/Chart.yaml @@ -0,0 +1,26 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/prometheus-community/helm-charts +apiVersion: v2 +appVersion: 2.13.0 +description: Install kube-state-metrics to generate and expose cluster-level metrics +home: https://github.com/kubernetes/kube-state-metrics/ +keywords: +- metric +- monitoring +- prometheus +- kubernetes +maintainers: +- email: tariq.ibrahim@mulesoft.com + name: tariq1890 +- email: manuel@rueg.eu + name: mrueg +- email: david@0xdc.me + name: dotdc +name: kube-state-metrics +sources: +- https://github.com/kubernetes/kube-state-metrics/ +type: application +version: 5.26.0 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/README.md b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/README.md new file mode 100644 index 000000000..843be89e6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/README.md @@ -0,0 +1,85 @@ +# kube-state-metrics Helm Chart + +Installs the [kube-state-metrics agent](https://github.com/kubernetes/kube-state-metrics). + +## Get Repository Info + +```console +helm repo add prometheus-community https://prometheus-community.github.io/helm-charts +helm repo update +``` + +_See [helm repo](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + + +## Install Chart + +```console +helm install [RELEASE_NAME] prometheus-community/kube-state-metrics [flags] +``` + +_See [configuration](#configuration) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Uninstall Chart + +```console +helm uninstall [RELEASE_NAME] +``` + +This removes all the Kubernetes components associated with the chart and deletes the release. + +_See [helm uninstall](https://helm.sh/docs/helm/helm_uninstall/) for command documentation._ + +## Upgrading Chart + +```console +helm upgrade [RELEASE_NAME] prometheus-community/kube-state-metrics [flags] +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +### Migrating from stable/kube-state-metrics and kubernetes/kube-state-metrics + +You can upgrade in-place: + +1. [get repository info](#get-repository-info) +1. [upgrade](#upgrading-chart) your existing release name using the new chart repository + +## Upgrading to v3.0.0 + +v3.0.0 includes kube-state-metrics v2.0, see the [changelog](https://github.com/kubernetes/kube-state-metrics/blob/release-2.0/CHANGELOG.md) for major changes on the application-side. + +The upgraded chart now the following changes: + +* Dropped support for helm v2 (helm v3 or later is required) +* collectors key was renamed to resources +* namespace key was renamed to namespaces + +## Configuration + +See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). To see all configurable options with detailed comments: + +```console +helm show values prometheus-community/kube-state-metrics +``` + +### kube-rbac-proxy + +You can enable `kube-state-metrics` endpoint protection using `kube-rbac-proxy`. By setting `kubeRBACProxy.enabled: true`, this chart will deploy one RBAC proxy container per endpoint (metrics & telemetry). +To authorize access, authenticate your requests (via a `ServiceAccount` for example) with a `ClusterRole` attached such as: + +```yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: kube-state-metrics-read +rules: + - apiGroups: [ "" ] + resources: ["services/kube-state-metrics"] + verbs: + - get +``` + +See [kube-rbac-proxy examples](https://github.com/brancz/kube-rbac-proxy/tree/master/examples/resource-attributes) for more details. diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt new file mode 100644 index 000000000..3589c24ec --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt @@ -0,0 +1,23 @@ +kube-state-metrics is a simple service that listens to the Kubernetes API server and generates metrics about the state of the objects. +The exposed metrics can be found here: +https://github.com/kubernetes/kube-state-metrics/blob/master/docs/README.md#exposed-metrics + +The metrics are exported on the HTTP endpoint /metrics on the listening port. +In your case, {{ template "kube-state-metrics.fullname" . }}.{{ template "kube-state-metrics.namespace" . }}.svc.cluster.local:{{ .Values.service.port }}/metrics + +They are served either as plaintext or protobuf depending on the Accept header. +They are designed to be consumed either by Prometheus itself or by a scraper that is compatible with scraping a Prometheus client endpoint. + +{{- if .Values.kubeRBACProxy.enabled}} + +kube-rbac-proxy endpoint protections is enabled: +- Metrics endpoints are now HTTPS +- Ensure that the client authenticates the requests (e.g. via service account) with the following role permissions: +``` +rules: + - apiGroups: [ "" ] + resources: ["services/{{ template "kube-state-metrics.fullname" . }}"] + verbs: + - get +``` +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl new file mode 100644 index 000000000..3dd326da4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl @@ -0,0 +1,156 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "kube-state-metrics.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 "kube-state-metrics.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 the name of the service account to use +*/}} +{{- define "kube-state-metrics.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "kube-state-metrics.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts +*/}} +{{- define "kube-state-metrics.namespace" -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "kube-state-metrics.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Generate basic labels +*/}} +{{- define "kube-state-metrics.labels" }} +helm.sh/chart: {{ template "kube-state-metrics.chart" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/component: metrics +app.kubernetes.io/part-of: {{ template "kube-state-metrics.name" . }} +{{- include "kube-state-metrics.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +{{- if .Values.customLabels }} +{{ tpl (toYaml .Values.customLabels) . }} +{{- end }} +{{- if .Values.releaseLabel }} +release: {{ .Release.Name }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "kube-state-metrics.selectorLabels" }} +{{- if .Values.selectorOverride }} +{{ toYaml .Values.selectorOverride }} +{{- else }} +app.kubernetes.io/name: {{ include "kube-state-metrics.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} +{{- end }} + +{{/* Sets default scrape limits for servicemonitor */}} +{{- define "servicemonitor.scrapeLimits" -}} +{{- with .sampleLimit }} +sampleLimit: {{ . }} +{{- end }} +{{- with .targetLimit }} +targetLimit: {{ . }} +{{- end }} +{{- with .labelLimit }} +labelLimit: {{ . }} +{{- end }} +{{- with .labelNameLengthLimit }} +labelNameLengthLimit: {{ . }} +{{- end }} +{{- with .labelValueLengthLimit }} +labelValueLengthLimit: {{ . }} +{{- end }} +{{- end -}} + +{{/* +Formats imagePullSecrets. Input is (dict "Values" .Values "imagePullSecrets" .{specific imagePullSecrets}) +*/}} +{{- define "kube-state-metrics.imagePullSecrets" -}} +{{- range (concat .Values.global.imagePullSecrets .imagePullSecrets) }} + {{- if eq (typeOf .) "map[string]interface {}" }} +- {{ toYaml . | trim }} + {{- else }} +- name: {{ . }} + {{- end }} +{{- end }} +{{- end -}} + +{{/* +The image to use for kube-state-metrics +*/}} +{{- define "kube-state-metrics.image" -}} +{{- if .Values.image.sha }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s@%s" .Values.global.imageRegistry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) .Values.image.sha }} +{{- else }} +{{- printf "%s/%s:%s@%s" .Values.image.registry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) .Values.image.sha }} +{{- end }} +{{- else }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s" .Values.global.imageRegistry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) }} +{{- else }} +{{- printf "%s/%s:%s" .Values.image.registry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +The image to use for kubeRBACProxy +*/}} +{{- define "kubeRBACProxy.image" -}} +{{- if .Values.kubeRBACProxy.image.sha }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s@%s" .Values.global.imageRegistry .Values.kubeRBACProxy.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.kubeRBACProxy.image.tag) .Values.kubeRBACProxy.image.sha }} +{{- else }} +{{- printf "%s/%s:%s@%s" .Values.kubeRBACProxy.image.registry .Values.kubeRBACProxy.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.kubeRBACProxy.image.tag) .Values.kubeRBACProxy.image.sha }} +{{- end }} +{{- else }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s" .Values.global.imageRegistry .Values.kubeRBACProxy.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.kubeRBACProxy.image.tag) }} +{{- else }} +{{- printf "%s/%s:%s" .Values.kubeRBACProxy.image.registry .Values.kubeRBACProxy.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.kubeRBACProxy.image.tag) }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml new file mode 100644 index 000000000..025cd47a8 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml @@ -0,0 +1,33 @@ +{{- if and .Values.networkPolicy.enabled (eq .Values.networkPolicy.flavor "cilium") }} +apiVersion: cilium.io/v2 +kind: CiliumNetworkPolicy +metadata: + {{- if .Values.annotations }} + annotations: + {{ toYaml .Values.annotations | nindent 4 }} + {{- end }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +spec: + endpointSelector: + matchLabels: + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + egress: + {{- if and .Values.networkPolicy.cilium .Values.networkPolicy.cilium.kubeApiServerSelector }} + {{ toYaml .Values.networkPolicy.cilium.kubeApiServerSelector | nindent 6 }} + {{- else }} + - toEntities: + - kube-apiserver + {{- end }} + ingress: + - toPorts: + - ports: + - port: {{ .Values.service.port | quote }} + protocol: TCP + {{- if .Values.selfMonitor.enabled }} + - port: {{ .Values.selfMonitor.telemetryPort | default 8081 | quote }} + protocol: TCP + {{ end }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..cf9f628d0 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.rbac.create .Values.rbac.useClusterRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole +{{- if .Values.rbac.useExistingRole }} + name: {{ .Values.rbac.useExistingRole }} +{{- else }} + name: {{ template "kube-state-metrics.fullname" . }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml new file mode 100644 index 000000000..d38a75a51 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml @@ -0,0 +1,16 @@ +{{- if .Values.customResourceState.enabled}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "kube-state-metrics.fullname" . }}-customresourcestate-config + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + {{- if .Values.annotations }} + annotations: + {{ toYaml .Values.annotations | nindent 4 }} + {{- end }} +data: + config.yaml: | + {{- toYaml .Values.customResourceState.config | nindent 4 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml new file mode 100644 index 000000000..2aff18888 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml @@ -0,0 +1,336 @@ +apiVersion: apps/v1 +{{- if .Values.autosharding.enabled }} +kind: StatefulSet +{{- else }} +kind: Deployment +{{- end }} +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + {{- if .Values.annotations }} + annotations: +{{ toYaml .Values.annotations | indent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + replicas: {{ .Values.replicas }} + {{- if not .Values.autosharding.enabled }} + strategy: + type: {{ .Values.updateStrategy | default "RollingUpdate" }} + {{- end }} + revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} + {{- if .Values.autosharding.enabled }} + serviceName: {{ template "kube-state-metrics.fullname" . }} + volumeClaimTemplates: [] + {{- end }} + template: + metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.podAnnotations }} + annotations: + {{ toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + spec: + automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} + hostNetwork: {{ .Values.hostNetwork }} + serviceAccountName: {{ template "kube-state-metrics.serviceAccountName" . }} + {{- if .Values.securityContext.enabled }} + securityContext: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.priorityClassName }} + priorityClassName: {{ .Values.priorityClassName }} + {{- end }} + {{- with .Values.initContainers }} + initContainers: + {{- toYaml . | nindent 6 }} + {{- end }} + containers: + {{- $servicePort := ternary 9090 (.Values.service.port | default 8080) .Values.kubeRBACProxy.enabled}} + {{- $telemetryPort := ternary 9091 (.Values.selfMonitor.telemetryPort | default 8081) .Values.kubeRBACProxy.enabled}} + - name: {{ template "kube-state-metrics.name" . }} + {{- if .Values.autosharding.enabled }} + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + {{- end }} + args: + {{- if .Values.extraArgs }} + {{- .Values.extraArgs | toYaml | nindent 8 }} + {{- end }} + - --port={{ $servicePort }} + {{- if .Values.collectors }} + - --resources={{ .Values.collectors | join "," }} + {{- end }} + {{- if .Values.metricLabelsAllowlist }} + - --metric-labels-allowlist={{ .Values.metricLabelsAllowlist | join "," }} + {{- end }} + {{- if .Values.metricAnnotationsAllowList }} + - --metric-annotations-allowlist={{ .Values.metricAnnotationsAllowList | join "," }} + {{- end }} + {{- if .Values.metricAllowlist }} + - --metric-allowlist={{ .Values.metricAllowlist | join "," }} + {{- end }} + {{- if .Values.metricDenylist }} + - --metric-denylist={{ .Values.metricDenylist | join "," }} + {{- end }} + {{- $namespaces := list }} + {{- if .Values.namespaces }} + {{- range $ns := join "," .Values.namespaces | split "," }} + {{- $namespaces = append $namespaces (tpl $ns $) }} + {{- end }} + {{- end }} + {{- if .Values.releaseNamespace }} + {{- $namespaces = append $namespaces ( include "kube-state-metrics.namespace" . ) }} + {{- end }} + {{- if $namespaces }} + - --namespaces={{ $namespaces | mustUniq | join "," }} + {{- end }} + {{- if .Values.namespacesDenylist }} + - --namespaces-denylist={{ tpl (.Values.namespacesDenylist | join ",") $ }} + {{- end }} + {{- if .Values.autosharding.enabled }} + - --pod=$(POD_NAME) + - --pod-namespace=$(POD_NAMESPACE) + {{- end }} + {{- if .Values.kubeconfig.enabled }} + - --kubeconfig=/opt/k8s/.kube/config + {{- end }} + {{- if .Values.kubeRBACProxy.enabled }} + - --telemetry-host=127.0.0.1 + - --telemetry-port={{ $telemetryPort }} + {{- else }} + {{- if .Values.selfMonitor.telemetryHost }} + - --telemetry-host={{ .Values.selfMonitor.telemetryHost }} + {{- end }} + {{- if .Values.selfMonitor.telemetryPort }} + - --telemetry-port={{ $telemetryPort }} + {{- end }} + {{- end }} + {{- if .Values.customResourceState.enabled }} + - --custom-resource-state-config-file=/etc/customresourcestate/config.yaml + {{- end }} + {{- if or (.Values.kubeconfig.enabled) (.Values.customResourceState.enabled) (.Values.volumeMounts) }} + volumeMounts: + {{- if .Values.kubeconfig.enabled }} + - name: kubeconfig + mountPath: /opt/k8s/.kube/ + readOnly: true + {{- end }} + {{- if .Values.customResourceState.enabled }} + - name: customresourcestate-config + mountPath: /etc/customresourcestate + readOnly: true + {{- end }} + {{- if .Values.volumeMounts }} +{{ toYaml .Values.volumeMounts | indent 8 }} + {{- end }} + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + image: {{ include "kube-state-metrics.image" . }} + {{- if eq .Values.kubeRBACProxy.enabled false }} + ports: + - containerPort: {{ .Values.service.port | default 8080}} + name: "http" + {{- if .Values.selfMonitor.enabled }} + - containerPort: {{ $telemetryPort }} + name: "metrics" + {{- end }} + {{- end }} + {{- if .Values.startupProbe.enabled }} + startupProbe: + failureThreshold: {{ .Values.startupProbe.failureThreshold }} + httpGet: + {{- if .Values.hostNetwork }} + host: 127.0.0.1 + {{- end }} + httpHeaders: + {{- range $_, $header := .Values.startupProbe.httpGet.httpHeaders }} + - name: {{ $header.name }} + value: {{ $header.value }} + {{- end }} + path: /healthz + port: {{ $servicePort }} + scheme: {{ upper .Values.startupProbe.httpGet.scheme }} + initialDelaySeconds: {{ .Values.startupProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.startupProbe.periodSeconds }} + successThreshold: {{ .Values.startupProbe.successThreshold }} + timeoutSeconds: {{ .Values.startupProbe.timeoutSeconds }} + {{- end }} + livenessProbe: + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + httpGet: + {{- if .Values.hostNetwork }} + host: 127.0.0.1 + {{- end }} + httpHeaders: + {{- range $_, $header := .Values.livenessProbe.httpGet.httpHeaders }} + - name: {{ $header.name }} + value: {{ $header.value }} + {{- end }} + path: /livez + port: {{ $servicePort }} + scheme: {{ upper .Values.livenessProbe.httpGet.scheme }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + readinessProbe: + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + httpGet: + {{- if .Values.hostNetwork }} + host: 127.0.0.1 + {{- end }} + httpHeaders: + {{- range $_, $header := .Values.readinessProbe.httpGet.httpHeaders }} + - name: {{ $header.name }} + value: {{ $header.value }} + {{- end }} + path: /readyz + port: {{ $servicePort }} + scheme: {{ upper .Values.readinessProbe.httpGet.scheme }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + resources: +{{ toYaml .Values.resources | indent 10 }} +{{- if .Values.containerSecurityContext }} + securityContext: +{{ toYaml .Values.containerSecurityContext | indent 10 }} +{{- end }} + {{- if .Values.kubeRBACProxy.enabled }} + - name: kube-rbac-proxy-http + args: + {{- if .Values.kubeRBACProxy.extraArgs }} + {{- .Values.kubeRBACProxy.extraArgs | toYaml | nindent 8 }} + {{- end }} + - --secure-listen-address=:{{ .Values.service.port | default 8080}} + - --upstream=http://127.0.0.1:{{ $servicePort }}/ + - --proxy-endpoints-port=8888 + - --config-file=/etc/kube-rbac-proxy-config/config-file.yaml + volumeMounts: + - name: kube-rbac-proxy-config + mountPath: /etc/kube-rbac-proxy-config + {{- with .Values.kubeRBACProxy.volumeMounts }} + {{- toYaml . | nindent 10 }} + {{- end }} + imagePullPolicy: {{ .Values.kubeRBACProxy.image.pullPolicy }} + image: {{ include "kubeRBACProxy.image" . }} + ports: + - containerPort: {{ .Values.service.port | default 8080}} + name: "http" + - containerPort: 8888 + name: "http-healthz" + readinessProbe: + httpGet: + scheme: HTTPS + port: 8888 + path: healthz + initialDelaySeconds: 5 + timeoutSeconds: 5 + {{- if .Values.kubeRBACProxy.resources }} + resources: +{{ toYaml .Values.kubeRBACProxy.resources | indent 10 }} +{{- end }} +{{- if .Values.kubeRBACProxy.containerSecurityContext }} + securityContext: +{{ toYaml .Values.kubeRBACProxy.containerSecurityContext | indent 10 }} +{{- end }} + {{- if .Values.selfMonitor.enabled }} + - name: kube-rbac-proxy-telemetry + args: + {{- if .Values.kubeRBACProxy.extraArgs }} + {{- .Values.kubeRBACProxy.extraArgs | toYaml | nindent 8 }} + {{- end }} + - --secure-listen-address=:{{ .Values.selfMonitor.telemetryPort | default 8081 }} + - --upstream=http://127.0.0.1:{{ $telemetryPort }}/ + - --proxy-endpoints-port=8889 + - --config-file=/etc/kube-rbac-proxy-config/config-file.yaml + volumeMounts: + - name: kube-rbac-proxy-config + mountPath: /etc/kube-rbac-proxy-config + {{- with .Values.kubeRBACProxy.volumeMounts }} + {{- toYaml . | nindent 10 }} + {{- end }} + imagePullPolicy: {{ .Values.kubeRBACProxy.image.pullPolicy }} + image: {{ include "kubeRBACProxy.image" . }} + ports: + - containerPort: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + name: "metrics" + - containerPort: 8889 + name: "metrics-healthz" + readinessProbe: + httpGet: + scheme: HTTPS + port: 8889 + path: healthz + initialDelaySeconds: 5 + timeoutSeconds: 5 + {{- if .Values.kubeRBACProxy.resources }} + resources: +{{ toYaml .Values.kubeRBACProxy.resources | indent 10 }} +{{- end }} +{{- if .Values.kubeRBACProxy.containerSecurityContext }} + securityContext: +{{ toYaml .Values.kubeRBACProxy.containerSecurityContext | indent 10 }} +{{- end }} + {{- end }} + {{- end }} + {{- with .Values.containers }} + {{- toYaml . | nindent 6 }} + {{- end }} +{{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} + imagePullSecrets: + {{- include "kube-state-metrics.imagePullSecrets" (dict "Values" .Values "imagePullSecrets" .Values.imagePullSecrets) | indent 8 }} + {{- end }} + {{- if .Values.affinity }} + affinity: +{{ toYaml .Values.affinity | indent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: +{{ tpl (toYaml .) $ | indent 8 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: +{{ tpl (toYaml .) $ | indent 8 }} + {{- end }} + {{- if .Values.topologySpreadConstraints }} + topologySpreadConstraints: +{{ toYaml .Values.topologySpreadConstraints | indent 8 }} + {{- end }} + {{- if or (.Values.kubeconfig.enabled) (.Values.customResourceState.enabled) (.Values.volumes) (.Values.kubeRBACProxy.enabled) }} + volumes: + {{- if .Values.kubeconfig.enabled}} + - name: kubeconfig + secret: + secretName: {{ template "kube-state-metrics.fullname" . }}-kubeconfig + {{- end }} + {{- if .Values.kubeRBACProxy.enabled}} + - name: kube-rbac-proxy-config + configMap: + name: {{ template "kube-state-metrics.fullname" . }}-rbac-config + {{- end }} + {{- if .Values.customResourceState.enabled}} + - name: customresourcestate-config + configMap: + name: {{ template "kube-state-metrics.fullname" . }}-customresourcestate-config + {{- end }} + {{- if .Values.volumes }} +{{ toYaml .Values.volumes | indent 8 }} + {{- end }} + {{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml new file mode 100644 index 000000000..567f7bf32 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraManifests }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml new file mode 100644 index 000000000..6af008450 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml @@ -0,0 +1,12 @@ +{{- if .Values.kubeconfig.enabled -}} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "kube-state-metrics.fullname" . }}-kubeconfig + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +type: Opaque +data: + config: '{{ .Values.kubeconfig.secret }}' +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml new file mode 100644 index 000000000..309b38ec5 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml @@ -0,0 +1,43 @@ +{{- if and .Values.networkPolicy.enabled (eq .Values.networkPolicy.flavor "kubernetes") }} +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + {{- if .Values.annotations }} + annotations: + {{ toYaml .Values.annotations | nindent 4 }} + {{- end }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +spec: + {{- if .Values.networkPolicy.egress }} + ## Deny all egress by default + egress: + {{- toYaml .Values.networkPolicy.egress | nindent 4 }} + {{- end }} + ingress: + {{- if .Values.networkPolicy.ingress }} + {{- toYaml .Values.networkPolicy.ingress | nindent 4 }} + {{- else }} + ## Allow ingress on default ports by default + - ports: + - port: {{ .Values.service.port | default 8080 }} + protocol: TCP + {{- if .Values.selfMonitor.enabled }} + {{- $telemetryPort := ternary 9091 (.Values.selfMonitor.telemetryPort | default 8081) .Values.kubeRBACProxy.enabled}} + - port: {{ $telemetryPort }} + protocol: TCP + {{- end }} + {{- end }} + podSelector: + {{- if .Values.networkPolicy.podSelector }} + {{- toYaml .Values.networkPolicy.podSelector | nindent 4 }} + {{- else }} + matchLabels: + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + {{- end }} + policyTypes: + - Ingress + - Egress +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml new file mode 100644 index 000000000..3771b511d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml @@ -0,0 +1,18 @@ +{{- if .Values.podDisruptionBudget -}} +{{ if $.Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" -}} +apiVersion: policy/v1 +{{- else -}} +apiVersion: policy/v1beta1 +{{- end }} +kind: PodDisruptionBudget +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +spec: + selector: + matchLabels: + app.kubernetes.io/name: {{ template "kube-state-metrics.name" . }} +{{ toYaml .Values.podDisruptionBudget | indent 2 }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..8905e113e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml @@ -0,0 +1,39 @@ +{{- if and .Values.podSecurityPolicy.enabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +{{- if .Values.podSecurityPolicy.annotations }} + annotations: +{{ toYaml .Values.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + volumes: + - 'secret' +{{- if .Values.podSecurityPolicy.additionalVolumes }} +{{ toYaml .Values.podSecurityPolicy.additionalVolumes | indent 4 }} +{{- end }} + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'MustRunAsNonRoot' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml new file mode 100644 index 000000000..654e4a3d5 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.podSecurityPolicy.enabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: psp-{{ template "kube-state-metrics.fullname" . }} +rules: +{{- $kubeTargetVersion := default .Capabilities.KubeVersion.GitVersion .Values.kubeTargetVersionOverride }} +{{- if semverCompare "> 1.15.0-0" $kubeTargetVersion }} +- apiGroups: ['policy'] +{{- else }} +- apiGroups: ['extensions'] +{{- end }} + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ template "kube-state-metrics.fullname" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml new file mode 100644 index 000000000..5b62a18bd --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.podSecurityPolicy.enabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: psp-{{ template "kube-state-metrics.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp-{{ template "kube-state-metrics.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml new file mode 100644 index 000000000..671dc9d66 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml @@ -0,0 +1,22 @@ +{{- if .Values.kubeRBACProxy.enabled}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "kube-state-metrics.fullname" . }}-rbac-config + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + {{- if .Values.annotations }} + annotations: + {{ toYaml .Values.annotations | nindent 4 }} + {{- end }} +data: + config-file.yaml: |+ + authorization: + resourceAttributes: + namespace: {{ template "kube-state-metrics.namespace" . }} + apiVersion: v1 + resource: services + subresource: {{ template "kube-state-metrics.fullname" . }} + name: {{ template "kube-state-metrics.fullname" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/role.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/role.yaml new file mode 100644 index 000000000..d33687f2d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/role.yaml @@ -0,0 +1,212 @@ +{{- if and (eq .Values.rbac.create true) (not .Values.rbac.useExistingRole) -}} +{{- range (ternary (join "," .Values.namespaces | split "," ) (list "") (eq $.Values.rbac.useClusterRole false)) }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +{{- if eq $.Values.rbac.useClusterRole false }} +kind: Role +{{- else }} +kind: ClusterRole +{{- end }} +metadata: + labels: + {{- include "kube-state-metrics.labels" $ | indent 4 }} + name: {{ template "kube-state-metrics.fullname" $ }} +{{- if eq $.Values.rbac.useClusterRole false }} + namespace: {{ . }} +{{- end }} +rules: +{{ if has "certificatesigningrequests" $.Values.collectors }} +- apiGroups: ["certificates.k8s.io"] + resources: + - certificatesigningrequests + verbs: ["list", "watch"] +{{ end -}} +{{ if has "configmaps" $.Values.collectors }} +- apiGroups: [""] + resources: + - configmaps + verbs: ["list", "watch"] +{{ end -}} +{{ if has "cronjobs" $.Values.collectors }} +- apiGroups: ["batch"] + resources: + - cronjobs + verbs: ["list", "watch"] +{{ end -}} +{{ if has "daemonsets" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - daemonsets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "deployments" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - deployments + verbs: ["list", "watch"] +{{ end -}} +{{ if has "endpoints" $.Values.collectors }} +- apiGroups: [""] + resources: + - endpoints + verbs: ["list", "watch"] +{{ end -}} +{{ if has "endpointslices" $.Values.collectors }} +- apiGroups: ["discovery.k8s.io"] + resources: + - endpointslices + verbs: ["list", "watch"] +{{ end -}} +{{ if has "horizontalpodautoscalers" $.Values.collectors }} +- apiGroups: ["autoscaling"] + resources: + - horizontalpodautoscalers + verbs: ["list", "watch"] +{{ end -}} +{{ if has "ingresses" $.Values.collectors }} +- apiGroups: ["extensions", "networking.k8s.io"] + resources: + - ingresses + verbs: ["list", "watch"] +{{ end -}} +{{ if has "jobs" $.Values.collectors }} +- apiGroups: ["batch"] + resources: + - jobs + verbs: ["list", "watch"] +{{ end -}} +{{ if has "leases" $.Values.collectors }} +- apiGroups: ["coordination.k8s.io"] + resources: + - leases + verbs: ["list", "watch"] +{{ end -}} +{{ if has "limitranges" $.Values.collectors }} +- apiGroups: [""] + resources: + - limitranges + verbs: ["list", "watch"] +{{ end -}} +{{ if has "mutatingwebhookconfigurations" $.Values.collectors }} +- apiGroups: ["admissionregistration.k8s.io"] + resources: + - mutatingwebhookconfigurations + verbs: ["list", "watch"] +{{ end -}} +{{ if has "namespaces" $.Values.collectors }} +- apiGroups: [""] + resources: + - namespaces + verbs: ["list", "watch"] +{{ end -}} +{{ if has "networkpolicies" $.Values.collectors }} +- apiGroups: ["networking.k8s.io"] + resources: + - networkpolicies + verbs: ["list", "watch"] +{{ end -}} +{{ if has "nodes" $.Values.collectors }} +- apiGroups: [""] + resources: + - nodes + verbs: ["list", "watch"] +{{ end -}} +{{ if has "persistentvolumeclaims" $.Values.collectors }} +- apiGroups: [""] + resources: + - persistentvolumeclaims + verbs: ["list", "watch"] +{{ end -}} +{{ if has "persistentvolumes" $.Values.collectors }} +- apiGroups: [""] + resources: + - persistentvolumes + verbs: ["list", "watch"] +{{ end -}} +{{ if has "poddisruptionbudgets" $.Values.collectors }} +- apiGroups: ["policy"] + resources: + - poddisruptionbudgets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "pods" $.Values.collectors }} +- apiGroups: [""] + resources: + - pods + verbs: ["list", "watch"] +{{ end -}} +{{ if has "replicasets" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - replicasets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "replicationcontrollers" $.Values.collectors }} +- apiGroups: [""] + resources: + - replicationcontrollers + verbs: ["list", "watch"] +{{ end -}} +{{ if has "resourcequotas" $.Values.collectors }} +- apiGroups: [""] + resources: + - resourcequotas + verbs: ["list", "watch"] +{{ end -}} +{{ if has "secrets" $.Values.collectors }} +- apiGroups: [""] + resources: + - secrets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "services" $.Values.collectors }} +- apiGroups: [""] + resources: + - services + verbs: ["list", "watch"] +{{ end -}} +{{ if has "statefulsets" $.Values.collectors }} +- apiGroups: ["apps"] + resources: + - statefulsets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "storageclasses" $.Values.collectors }} +- apiGroups: ["storage.k8s.io"] + resources: + - storageclasses + verbs: ["list", "watch"] +{{ end -}} +{{ if has "validatingwebhookconfigurations" $.Values.collectors }} +- apiGroups: ["admissionregistration.k8s.io"] + resources: + - validatingwebhookconfigurations + verbs: ["list", "watch"] +{{ end -}} +{{ if has "volumeattachments" $.Values.collectors }} +- apiGroups: ["storage.k8s.io"] + resources: + - volumeattachments + verbs: ["list", "watch"] +{{ end -}} +{{- if $.Values.kubeRBACProxy.enabled }} +- apiGroups: ["authentication.k8s.io"] + resources: + - tokenreviews + verbs: ["create"] +- apiGroups: ["authorization.k8s.io"] + resources: + - subjectaccessreviews + verbs: ["create"] +{{- end }} +{{- if $.Values.customResourceState.enabled }} +- apiGroups: ["apiextensions.k8s.io"] + resources: + - customresourcedefinitions + verbs: ["list", "watch"] +{{- end }} +{{ if $.Values.rbac.extraRules }} +{{ toYaml $.Values.rbac.extraRules }} +{{ end }} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml new file mode 100644 index 000000000..330651b73 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml @@ -0,0 +1,24 @@ +{{- if and (eq .Values.rbac.create true) (eq .Values.rbac.useClusterRole false) -}} +{{- range (join "," $.Values.namespaces) | split "," }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" $ | indent 4 }} + name: {{ template "kube-state-metrics.fullname" $ }} + namespace: {{ . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role +{{- if (not $.Values.rbac.useExistingRole) }} + name: {{ template "kube-state-metrics.fullname" $ }} +{{- else }} + name: {{ $.Values.rbac.useExistingRole }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" $ }} + namespace: {{ template "kube-state-metrics.namespace" $ }} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/service.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/service.yaml new file mode 100644 index 000000000..90c235148 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/service.yaml @@ -0,0 +1,53 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + annotations: + {{- if .Values.prometheusScrape }} + prometheus.io/scrape: '{{ .Values.prometheusScrape }}' + {{- end }} + {{- if .Values.service.annotations }} + {{- toYaml .Values.service.annotations | nindent 4 }} + {{- end }} +spec: + type: "{{ .Values.service.type }}" + {{- if .Values.service.ipDualStack.enabled }} + ipFamilies: {{ toYaml .Values.service.ipDualStack.ipFamilies | nindent 4 }} + ipFamilyPolicy: {{ .Values.service.ipDualStack.ipFamilyPolicy }} + {{- end }} + ports: + - name: "http" + protocol: TCP + port: {{ .Values.service.port | default 8080}} + {{- if .Values.service.nodePort }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + targetPort: {{ .Values.service.port | default 8080}} + {{ if .Values.selfMonitor.enabled }} + - name: "metrics" + protocol: TCP + port: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + targetPort: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + {{- if .Values.selfMonitor.telemetryNodePort }} + nodePort: {{ .Values.selfMonitor.telemetryNodePort }} + {{- end }} + {{ end }} +{{- if .Values.service.loadBalancerIP }} + loadBalancerIP: "{{ .Values.service.loadBalancerIP }}" +{{- end }} +{{- if .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} +{{- if .Values.autosharding.enabled }} + clusterIP: None +{{- else if .Values.service.clusterIP }} + clusterIP: "{{ .Values.service.clusterIP }}" +{{- end }} + selector: + {{- include "kube-state-metrics.selectorLabels" . | indent 4 }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml new file mode 100644 index 000000000..c302bc7ca --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml @@ -0,0 +1,18 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- if .Values.serviceAccount.annotations }} + annotations: +{{ toYaml .Values.serviceAccount.annotations | indent 4 }} +{{- end }} +{{- if or .Values.serviceAccount.imagePullSecrets .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- include "kube-state-metrics.imagePullSecrets" (dict "Values" .Values "imagePullSecrets" .Values.serviceAccount.imagePullSecrets) | indent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml new file mode 100644 index 000000000..99d7fa924 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml @@ -0,0 +1,120 @@ +{{- if .Values.prometheus.monitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + {{- with .Values.prometheus.monitor.additionalLabels }} + {{- tpl (toYaml . | nindent 4) $ }} + {{- end }} + {{- with .Values.prometheus.monitor.annotations }} + annotations: + {{- tpl (toYaml . | nindent 4) $ }} + {{- end }} +spec: + jobLabel: {{ default "app.kubernetes.io/name" .Values.prometheus.monitor.jobLabel }} + {{- with .Values.prometheus.monitor.targetLabels }} + targetLabels: + {{- toYaml . | trim | nindent 4 }} + {{- end }} + {{- with .Values.prometheus.monitor.podTargetLabels }} + podTargetLabels: + {{- toYaml . | trim | nindent 4 }} + {{- end }} + {{- include "servicemonitor.scrapeLimits" .Values.prometheus.monitor | indent 2 }} + {{- if .Values.prometheus.monitor.namespaceSelector }} + namespaceSelector: + matchNames: + {{- with .Values.prometheus.monitor.namespaceSelector }} + {{- toYaml . | nindent 6 }} + {{- end }} + {{- end }} + selector: + matchLabels: + {{- with .Values.prometheus.monitor.selectorOverride }} + {{- toYaml . | nindent 6 }} + {{- else }} + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + {{- end }} + endpoints: + - port: http + {{- if or .Values.prometheus.monitor.http.interval .Values.prometheus.monitor.interval }} + interval: {{ .Values.prometheus.monitor.http.interval | default .Values.prometheus.monitor.interval }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.scrapeTimeout .Values.prometheus.monitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.prometheus.monitor.http.scrapeTimeout | default .Values.prometheus.monitor.scrapeTimeout }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.proxyUrl .Values.prometheus.monitor.proxyUrl }} + proxyUrl: {{ .Values.prometheus.monitor.http.proxyUrl | default .Values.prometheus.monitor.proxyUrl }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.enableHttp2 .Values.prometheus.monitor.enableHttp2 }} + enableHttp2: {{ .Values.prometheus.monitor.http.enableHttp2 | default .Values.prometheus.monitor.enableHttp2 }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.honorLabels .Values.prometheus.monitor.honorLabels }} + honorLabels: true + {{- end }} + {{- if or .Values.prometheus.monitor.http.metricRelabelings .Values.prometheus.monitor.metricRelabelings }} + metricRelabelings: + {{- toYaml (.Values.prometheus.monitor.http.metricRelabelings | default .Values.prometheus.monitor.metricRelabelings) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.relabelings .Values.prometheus.monitor.relabelings }} + relabelings: + {{- toYaml (.Values.prometheus.monitor.http.relabelings | default .Values.prometheus.monitor.relabelings) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.scheme .Values.prometheus.monitor.scheme }} + scheme: {{ .Values.prometheus.monitor.http.scheme | default .Values.prometheus.monitor.scheme }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.tlsConfig .Values.prometheus.monitor.tlsConfig }} + tlsConfig: + {{- toYaml (.Values.prometheus.monitor.http.tlsConfig | default .Values.prometheus.monitor.tlsConfig) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.http.bearerTokenFile .Values.prometheus.monitor.bearerTokenFile }} + bearerTokenFile: {{ .Values.prometheus.monitor.http.bearerTokenFile | default .Values.prometheus.monitor.bearerTokenFile }} + {{- end }} + {{- with (.Values.prometheus.monitor.http.bearerTokenSecret | default .Values.prometheus.monitor.bearerTokenSecret) }} + bearerTokenSecret: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.selfMonitor.enabled }} + - port: metrics + {{- if or .Values.prometheus.monitor.metrics.interval .Values.prometheus.monitor.interval }} + interval: {{ .Values.prometheus.monitor.metrics.interval | default .Values.prometheus.monitor.interval }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.scrapeTimeout .Values.prometheus.monitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.prometheus.monitor.metrics.scrapeTimeout | default .Values.prometheus.monitor.scrapeTimeout }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.proxyUrl .Values.prometheus.monitor.proxyUrl }} + proxyUrl: {{ .Values.prometheus.monitor.metrics.proxyUrl | default .Values.prometheus.monitor.proxyUrl }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.enableHttp2 .Values.prometheus.monitor.enableHttp2 }} + enableHttp2: {{ .Values.prometheus.monitor.metrics.enableHttp2 | default .Values.prometheus.monitor.enableHttp2 }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.honorLabels .Values.prometheus.monitor.honorLabels }} + honorLabels: true + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.metricRelabelings .Values.prometheus.monitor.metricRelabelings }} + metricRelabelings: + {{- toYaml (.Values.prometheus.monitor.metrics.metricRelabelings | default .Values.prometheus.monitor.metricRelabelings) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.relabelings .Values.prometheus.monitor.relabelings }} + relabelings: + {{- toYaml (.Values.prometheus.monitor.metrics.relabelings | default .Values.prometheus.monitor.relabelings) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.scheme .Values.prometheus.monitor.scheme }} + scheme: {{ .Values.prometheus.monitor.metrics.scheme | default .Values.prometheus.monitor.scheme }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.tlsConfig .Values.prometheus.monitor.tlsConfig }} + tlsConfig: + {{- toYaml (.Values.prometheus.monitor.metrics.tlsConfig | default .Values.prometheus.monitor.tlsConfig) | nindent 8 }} + {{- end }} + {{- if or .Values.prometheus.monitor.metrics.bearerTokenFile .Values.prometheus.monitor.bearerTokenFile }} + bearerTokenFile: {{ .Values.prometheus.monitor.metrics.bearerTokenFile | default .Values.prometheus.monitor.bearerTokenFile }} + {{- end }} + {{- with (.Values.prometheus.monitor.metrics.bearerTokenSecret | default .Values.prometheus.monitor.bearerTokenSecret) }} + bearerTokenSecret: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml new file mode 100644 index 000000000..489de147c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml @@ -0,0 +1,26 @@ +{{- if and .Values.autosharding.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - get +- apiGroups: + - apps + resourceNames: + - {{ template "kube-state-metrics.fullname" . }} + resources: + - statefulsets + verbs: + - get + - list + - watch +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml new file mode 100644 index 000000000..73b37a4f6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml @@ -0,0 +1,17 @@ +{{- if and .Values.autosharding.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml new file mode 100644 index 000000000..f46305b51 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml @@ -0,0 +1,44 @@ +{{- if and (.Capabilities.APIVersions.Has "autoscaling.k8s.io/v1") (.Values.verticalPodAutoscaler.enabled) }} +apiVersion: autoscaling.k8s.io/v1 +kind: VerticalPodAutoscaler +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +spec: + {{- with .Values.verticalPodAutoscaler.recommenders }} + recommenders: + {{- toYaml . | nindent 4 }} + {{- end }} + resourcePolicy: + containerPolicies: + - containerName: {{ template "kube-state-metrics.name" . }} + {{- with .Values.verticalPodAutoscaler.controlledResources }} + controlledResources: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.verticalPodAutoscaler.controlledValues }} + controlledValues: {{ .Values.verticalPodAutoscaler.controlledValues }} + {{- end }} + {{- if .Values.verticalPodAutoscaler.maxAllowed }} + maxAllowed: + {{ toYaml .Values.verticalPodAutoscaler.maxAllowed | nindent 8 }} + {{- end }} + {{- if .Values.verticalPodAutoscaler.minAllowed }} + minAllowed: + {{ toYaml .Values.verticalPodAutoscaler.minAllowed | nindent 8 }} + {{- end }} + targetRef: + apiVersion: apps/v1 + {{- if .Values.autosharding.enabled }} + kind: StatefulSet + {{- else }} + kind: Deployment + {{- end }} + name: {{ template "kube-state-metrics.fullname" . }} + {{- with .Values.verticalPodAutoscaler.updatePolicy }} + updatePolicy: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/values.yaml new file mode 100644 index 000000000..a7b2bdad6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/kube-state-metrics/values.yaml @@ -0,0 +1,542 @@ +# Default values for kube-state-metrics. +prometheusScrape: true +image: + registry: registry.k8s.io + repository: kube-state-metrics/kube-state-metrics + # If unset use v + .Charts.appVersion + tag: "" + sha: "" + pullPolicy: IfNotPresent + +imagePullSecrets: [] +# - name: "image-pull-secret" + +global: + # To help compatibility with other charts which use global.imagePullSecrets. + # Allow either an array of {name: pullSecret} maps (k8s-style), or an array of strings (more common helm-style). + # global: + # imagePullSecrets: + # - name: pullSecret1 + # - name: pullSecret2 + # or + # global: + # imagePullSecrets: + # - pullSecret1 + # - pullSecret2 + imagePullSecrets: [] + # + # Allow parent charts to override registry hostname + imageRegistry: "" + +# If set to true, this will deploy kube-state-metrics as a StatefulSet and the data +# will be automatically sharded across <.Values.replicas> pods using the built-in +# autodiscovery feature: https://github.com/kubernetes/kube-state-metrics#automated-sharding +# This is an experimental feature and there are no stability guarantees. +autosharding: + enabled: false + +replicas: 1 + +# Change the deployment strategy when autosharding is disabled. +# ref: https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#strategy +# The default is "RollingUpdate" as per Kubernetes defaults. +# During a release, 'RollingUpdate' can lead to two running instances for a short period of time while 'Recreate' can create a small gap in data. +# updateStrategy: Recreate + +# Number of old history to retain to allow rollback +# Default Kubernetes value is set to 10 +revisionHistoryLimit: 10 + +# List of additional cli arguments to configure kube-state-metrics +# for example: --enable-gzip-encoding, --log-file, etc. +# all the possible args can be found here: https://github.com/kubernetes/kube-state-metrics/blob/master/docs/cli-arguments.md +extraArgs: [] + +# If false then the user will opt out of automounting API credentials. +automountServiceAccountToken: true + +service: + port: 8080 + # Default to clusterIP for backward compatibility + type: ClusterIP + ipDualStack: + enabled: false + ipFamilies: ["IPv6", "IPv4"] + ipFamilyPolicy: "PreferDualStack" + nodePort: 0 + loadBalancerIP: "" + # Only allow access to the loadBalancerIP from these IPs + loadBalancerSourceRanges: [] + clusterIP: "" + annotations: {} + +## Additional labels to add to all resources +customLabels: {} + # app: kube-state-metrics + +## Override selector labels +selectorOverride: {} + +## set to true to add the release label so scraping of the servicemonitor with kube-prometheus-stack works out of the box +releaseLabel: false + +hostNetwork: false + +rbac: + # If true, create & use RBAC resources + create: true + + # Set to a rolename to use existing role - skipping role creating - but still doing serviceaccount and rolebinding to it, rolename set here. + # useExistingRole: your-existing-role + + # If set to false - Run without Cluteradmin privs needed - ONLY works if namespace is also set (if useExistingRole is set this name is used as ClusterRole or Role to bind to) + useClusterRole: true + + # Add permissions for CustomResources' apiGroups in Role/ClusterRole. Should be used in conjunction with Custom Resource State Metrics configuration + # Example: + # - apiGroups: ["monitoring.coreos.com"] + # resources: ["prometheuses"] + # verbs: ["list", "watch"] + extraRules: [] + +# Configure kube-rbac-proxy. When enabled, creates one kube-rbac-proxy container per exposed HTTP endpoint (metrics and telemetry if enabled). +# The requests are served through the same service but requests are then HTTPS. +kubeRBACProxy: + enabled: false + image: + registry: quay.io + repository: brancz/kube-rbac-proxy + tag: v0.18.0 + sha: "" + pullPolicy: IfNotPresent + + # List of additional cli arguments to configure kube-rbac-prxy + # for example: --tls-cipher-suites, --log-file, etc. + # all the possible args can be found here: https://github.com/brancz/kube-rbac-proxy#usage + extraArgs: [] + + ## Specify security settings for a Container + ## Allows overrides and additional options compared to (Pod) securityContext + ## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container + containerSecurityContext: + readOnlyRootFilesystem: true + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + + 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: 64Mi + # requests: + # cpu: 10m + # memory: 32Mi + + ## volumeMounts enables mounting custom volumes in rbac-proxy containers + ## Useful for TLS certificates and keys + volumeMounts: [] + # - mountPath: /etc/tls + # name: kube-rbac-proxy-tls + # readOnly: true + +serviceAccount: + # Specifies whether a ServiceAccount should be created, require rbac true + 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: + # Reference to one or more secrets to be used when pulling images + # ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + imagePullSecrets: [] + # ServiceAccount annotations. + # Use case: AWS EKS IAM roles for service accounts + # ref: https://docs.aws.amazon.com/eks/latest/userguide/specify-service-account-role.html + annotations: {} + # If false then the user will opt out of automounting API credentials. + automountServiceAccountToken: true + +prometheus: + monitor: + enabled: false + annotations: {} + additionalLabels: {} + namespace: "" + namespaceSelector: [] + jobLabel: "" + targetLabels: [] + podTargetLabels: [] + ## SampleLimit defines per-scrape limit on number of scraped samples that will be accepted. + ## + sampleLimit: 0 + + ## TargetLimit defines a limit on the number of scraped targets that will be accepted. + ## + targetLimit: 0 + + ## Per-scrape limit on number of labels that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelLimit: 0 + + ## Per-scrape limit on length of labels name that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelNameLengthLimit: 0 + + ## Per-scrape limit on length of labels value that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelValueLengthLimit: 0 + selectorOverride: {} + + ## kube-state-metrics endpoint + http: + interval: "" + scrapeTimeout: "" + proxyUrl: "" + ## Whether to enable HTTP2 for servicemonitor + enableHttp2: false + honorLabels: false + metricRelabelings: [] + relabelings: [] + scheme: "" + ## File to read bearer token for scraping targets + bearerTokenFile: "" + ## Secret to mount to read bearer token for scraping targets. The secret needs + ## to be in the same namespace as the service monitor and accessible by the + ## Prometheus Operator + bearerTokenSecret: {} + # name: secret-name + # key: key-name + tlsConfig: {} + + ## selfMonitor endpoint + metrics: + interval: "" + scrapeTimeout: "" + proxyUrl: "" + ## Whether to enable HTTP2 for servicemonitor + enableHttp2: false + honorLabels: false + metricRelabelings: [] + relabelings: [] + scheme: "" + ## File to read bearer token for scraping targets + bearerTokenFile: "" + ## Secret to mount to read bearer token for scraping targets. The secret needs + ## to be in the same namespace as the service monitor and accessible by the + ## Prometheus Operator + bearerTokenSecret: {} + # name: secret-name + # key: key-name + tlsConfig: {} + +## Specify if a Pod Security Policy for kube-state-metrics must be created +## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ +## +podSecurityPolicy: + enabled: false + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + additionalVolumes: [] + +## Configure network policy for kube-state-metrics +networkPolicy: + enabled: false + # networkPolicy.flavor -- Flavor of the network policy to use. + # Can be: + # * kubernetes for networking.k8s.io/v1/NetworkPolicy + # * cilium for cilium.io/v2/CiliumNetworkPolicy + flavor: kubernetes + + ## Configure the cilium network policy kube-apiserver selector + # cilium: + # kubeApiServerSelector: + # - toEntities: + # - kube-apiserver + + # egress: + # - {} + # ingress: + # - {} + # podSelector: + # matchLabels: + # app.kubernetes.io/name: kube-state-metrics + +securityContext: + enabled: true + runAsGroup: 65534 + runAsUser: 65534 + fsGroup: 65534 + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + +## Specify security settings for a Container +## Allows overrides and additional options compared to (Pod) securityContext +## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container +containerSecurityContext: + readOnlyRootFilesystem: true + allowPrivilegeEscalation: false + capabilities: + drop: + - ALL + +## Node labels for pod assignment +## Ref: https://kubernetes.io/docs/user-guide/node-selection/ +nodeSelector: {} + +## Affinity settings for pod assignment +## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ +affinity: {} + +## Tolerations for pod assignment +## Ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +tolerations: [] + +## Topology spread constraints for pod assignment +## Ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ +topologySpreadConstraints: [] + +# Annotations to be added to the deployment/statefulset +annotations: {} + +# Annotations to be added to the pod +podAnnotations: {} + +# Labels to be added to the pod +podLabels: {} + +## Assign a PriorityClassName to pods if set +# priorityClassName: "" + +# Ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ +podDisruptionBudget: {} + +# Comma-separated list of metrics to be exposed. +# This list comprises of exact metric names and/or regex patterns. +# The allowlist and denylist are mutually exclusive. +metricAllowlist: [] + +# Comma-separated list of metrics not to be enabled. +# This list comprises of exact metric names and/or regex patterns. +# The allowlist and denylist are mutually exclusive. +metricDenylist: [] + +# Comma-separated list of additional Kubernetes label keys that will be used in the resource's +# labels metric. By default the metric contains only name and namespace labels. +# To include additional labels, provide a list of resource names in their plural form and Kubernetes +# label keys you would like to allow for them (Example: '=namespaces=[k8s-label-1,k8s-label-n,...],pods=[app],...)'. +# A single '*' can be provided per resource instead to allow any labels, but that has +# severe performance implications (Example: '=pods=[*]'). +metricLabelsAllowlist: [] + # - namespaces=[k8s-label-1,k8s-label-n] + +# Comma-separated list of Kubernetes annotations keys that will be used in the resource' +# labels metric. By default the metric contains only name and namespace labels. +# To include additional annotations provide a list of resource names in their plural form and Kubernetes +# annotation keys you would like to allow for them (Example: '=namespaces=[kubernetes.io/team,...],pods=[kubernetes.io/team],...)'. +# A single '*' can be provided per resource instead to allow any annotations, but that has +# severe performance implications (Example: '=pods=[*]'). +metricAnnotationsAllowList: [] + # - pods=[k8s-annotation-1,k8s-annotation-n] + +# Available collectors for kube-state-metrics. +# By default, all available resources are enabled, comment out to disable. +collectors: + - certificatesigningrequests + - configmaps + - cronjobs + - daemonsets + - deployments + - endpoints + - horizontalpodautoscalers + - ingresses + - jobs + - leases + - limitranges + - mutatingwebhookconfigurations + - namespaces + - networkpolicies + - nodes + - persistentvolumeclaims + - persistentvolumes + - poddisruptionbudgets + - pods + - replicasets + - replicationcontrollers + - resourcequotas + - secrets + - services + - statefulsets + - storageclasses + - validatingwebhookconfigurations + - volumeattachments + +# Enabling kubeconfig will pass the --kubeconfig argument to the container +kubeconfig: + enabled: false + # base64 encoded kube-config file + secret: + +# Enabling support for customResourceState, will create a configMap including your config that will be read from kube-state-metrics +customResourceState: + enabled: false + # Add (Cluster)Role permissions to list/watch the customResources defined in the config to rbac.extraRules + config: {} + +# Enable only the release namespace for collecting resources. By default all namespaces are collected. +# If releaseNamespace and namespaces are both set a merged list will be collected. +releaseNamespace: false + +# Comma-separated list(string) or yaml list of namespaces to be enabled for collecting resources. By default all namespaces are collected. +namespaces: "" + +# Comma-separated list of namespaces not to be enabled. If namespaces and namespaces-denylist are both set, +# only namespaces that are excluded in namespaces-denylist will be used. +namespacesDenylist: "" + +## Override the deployment namespace +## +namespaceOverride: "" + +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: 64Mi + # requests: + # cpu: 10m + # memory: 32Mi + +## Provide a k8s version to define apiGroups for podSecurityPolicy Cluster Role. +## For example: kubeTargetVersionOverride: 1.14.9 +## +kubeTargetVersionOverride: "" + +# Enable self metrics configuration for service and Service Monitor +# Default values for telemetry configuration can be overridden +# If you set telemetryNodePort, you must also set service.type to NodePort +selfMonitor: + enabled: false + # telemetryHost: 0.0.0.0 + # telemetryPort: 8081 + # telemetryNodePort: 0 + +# Enable vertical pod autoscaler support for kube-state-metrics +verticalPodAutoscaler: + enabled: false + + # Recommender responsible for generating recommendation for the object. + # List should be empty (then the default recommender will generate the recommendation) + # or contain exactly one recommender. + # recommenders: [] + # - name: custom-recommender-performance + + # List of resources that the vertical pod autoscaler can control. Defaults to cpu and memory + controlledResources: [] + # Specifies which resource values should be controlled: RequestsOnly or RequestsAndLimits. + # controlledValues: RequestsAndLimits + + # Define the max allowed resources for the pod + maxAllowed: {} + # cpu: 200m + # memory: 100Mi + # Define the min allowed resources for the pod + minAllowed: {} + # cpu: 200m + # memory: 100Mi + + # updatePolicy: + # Specifies minimal number of replicas which need to be alive for VPA Updater to attempt pod eviction + # minReplicas: 1 + # Specifies whether recommended updates are applied when a Pod is started and whether recommended updates + # are applied during the life of a Pod. Possible values are "Off", "Initial", "Recreate", and "Auto". + # updateMode: Auto + +# volumeMounts are used to add custom volume mounts to deployment. +# See example below +volumeMounts: [] +# - mountPath: /etc/config +# name: config-volume + +# volumes are used to add custom volumes to deployment +# See example below +volumes: [] +# - configMap: +# name: cm-for-volume +# name: config-volume + +# Extra manifests to deploy as an array +extraManifests: [] + # - apiVersion: v1 + # kind: ConfigMap + # metadata: + # labels: + # name: prometheus-extra + # data: + # extra-data: "value" + +## Containers allows injecting additional containers. +containers: [] + # - name: crd-init + # image: kiwigrid/k8s-sidecar:latest + +## InitContainers allows injecting additional initContainers. +initContainers: [] + # - name: crd-sidecar + # image: kiwigrid/k8s-sidecar:latest + +## Settings for startup, liveness and readiness probes +## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/ +## + +## Startup probe can optionally be enabled. +## +startupProbe: + enabled: false + failureThreshold: 3 + httpGet: + httpHeaders: [] + scheme: http + initialDelaySeconds: 0 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 5 + +## Liveness probe +## +livenessProbe: + failureThreshold: 3 + httpGet: + httpHeaders: [] + scheme: http + initialDelaySeconds: 5 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 5 + +## Readiness probe +## +readinessProbe: + failureThreshold: 3 + httpGet: + httpHeaders: [] + scheme: http + initialDelaySeconds: 5 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 5 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/.helmignore b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/.helmignore new file mode 100644 index 000000000..f0c131944 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/.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/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml new file mode 100644 index 000000000..626592ca5 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml @@ -0,0 +1,25 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/prometheus-community/helm-charts +apiVersion: v2 +appVersion: 1.8.2 +description: A Helm chart for prometheus node-exporter +home: https://github.com/prometheus/node_exporter/ +keywords: +- node-exporter +- prometheus +- exporter +maintainers: +- email: gianrubio@gmail.com + name: gianrubio +- email: zanhsieh@gmail.com + name: zanhsieh +- email: rootsandtrees@posteo.de + name: zeritti +name: prometheus-node-exporter +sources: +- https://github.com/prometheus/node_exporter/ +type: application +version: 4.40.0 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/README.md b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/README.md new file mode 100644 index 000000000..ef8384410 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/README.md @@ -0,0 +1,96 @@ +# Prometheus Node Exporter + +Prometheus exporter for hardware and OS metrics exposed by *NIX kernels, written in Go with pluggable metric collectors. + +This chart bootstraps a Prometheus [Node Exporter](http://github.com/prometheus/node_exporter) daemonset on a [Kubernetes](http://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. + +## Get Repository Info + +```console +helm repo add prometheus-community https://prometheus-community.github.io/helm-charts +helm repo update +``` + +_See [helm repo](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Install Chart + +```console +helm install [RELEASE_NAME] prometheus-community/prometheus-node-exporter +``` + +_See [configuration](#configuring) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Uninstall Chart + +```console +helm uninstall [RELEASE_NAME] +``` + +This removes all the Kubernetes components associated with the chart and deletes the release. + +_See [helm uninstall](https://helm.sh/docs/helm/helm_uninstall/) for command documentation._ + +## Upgrading Chart + +```console +helm upgrade [RELEASE_NAME] prometheus-community/prometheus-node-exporter --install +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +### 3.x to 4.x + +Starting from version 4.0.0, the `node exporter` chart is using the [Kubernetes recommended labels](https://kubernetes.io/docs/concepts/overview/working-with-objects/common-labels/). Therefore you have to delete the daemonset before you upgrade. + +```console +kubectl delete daemonset -l app=prometheus-node-exporter +helm upgrade -i prometheus-node-exporter prometheus-community/prometheus-node-exporter +``` + +If you use your own custom [ServiceMonitor](https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#servicemonitor) or [PodMonitor](https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#podmonitor), please ensure to upgrade their `selector` fields accordingly to the new labels. + +### From 2.x to 3.x + +Change the following: + +```yaml +hostRootFsMount: true +``` + +to: + +```yaml +hostRootFsMount: + enabled: true + mountPropagation: HostToContainer +``` + +## Configuring + +See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). To see all configurable options with detailed comments, visit the chart's [values.yaml](./values.yaml), or run these configuration commands: + +```console +helm show values prometheus-community/prometheus-node-exporter +``` + +### kube-rbac-proxy + +You can enable `prometheus-node-exporter` endpoint protection using `kube-rbac-proxy`. By setting `kubeRBACProxy.enabled: true`, this chart will deploy a RBAC proxy container protecting the node-exporter endpoint. +To authorize access, authenticate your requests (via a `ServiceAccount` for example) with a `ClusterRole` attached such as: + +```yaml +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: prometheus-node-exporter-read +rules: + - apiGroups: [ "" ] + resources: ["services/node-exporter-prometheus-node-exporter"] + verbs: + - get +``` + +See [kube-rbac-proxy examples](https://github.com/brancz/kube-rbac-proxy/tree/master/examples/resource-attributes) for more details. diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/common-labels-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/common-labels-values.yaml new file mode 100644 index 000000000..719e9356e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/common-labels-values.yaml @@ -0,0 +1,4 @@ +--- +commonLabels: + foo: bar + baz: '{{ include "prometheus-node-exporter.fullname" . }}' diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/default-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/default-values.yaml new file mode 100644 index 000000000..39d98f716 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/default-values.yaml @@ -0,0 +1 @@ +## Default values test case diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/networkpolicy-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/networkpolicy-values.yaml new file mode 100644 index 000000000..bcea8de49 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/networkpolicy-values.yaml @@ -0,0 +1,5 @@ +networkPolicy: + enabled: true + ingress: + - ports: + - port: 9100 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/pod-labels-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/pod-labels-values.yaml new file mode 100644 index 000000000..7de36a6ab --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/pod-labels-values.yaml @@ -0,0 +1,4 @@ +--- +podLabels: + foo: bar + baz: '{{ .Chart.AppVersion }}' diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml new file mode 100644 index 000000000..dbfb4b67f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml @@ -0,0 +1,3 @@ +service: + targetPort: 9102 + port: 9102 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/service-labels-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/service-labels-values.yaml new file mode 100644 index 000000000..9c5e36506 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/service-labels-values.yaml @@ -0,0 +1,5 @@ +--- +service: + labels: + foo: bar + baz: quux diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/serviceport-values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/serviceport-values.yaml new file mode 100644 index 000000000..b0b7be656 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/ci/serviceport-values.yaml @@ -0,0 +1,3 @@ +--- +service: + servicePort: 80 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt new file mode 100644 index 000000000..db8584def --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt @@ -0,0 +1,29 @@ +1. Get the application URL by running these commands: +{{- if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ template "prometheus-node-exporter.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus-node-exporter.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ template "prometheus-node-exporter.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc -w {{ template "prometheus-node-exporter.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ template "prometheus-node-exporter.namespace" . }} {{ template "prometheus-node-exporter.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.service.port }} +{{- else if contains "ClusterIP" .Values.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ template "prometheus-node-exporter.namespace" . }} -l "app.kubernetes.io/name={{ template "prometheus-node-exporter.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + echo "Visit http://127.0.0.1:9100 to use your application" + kubectl port-forward --namespace {{ template "prometheus-node-exporter.namespace" . }} $POD_NAME 9100 +{{- end }} + +{{- if .Values.kubeRBACProxy.enabled}} + +kube-rbac-proxy endpoint protections is enabled: +- Metrics endpoints is now HTTPS +- Ensure that the client authenticates the requests (e.g. via service account) with the following role permissions: +``` +rules: + - apiGroups: [ "" ] + resources: ["services/{{ template "prometheus-node-exporter.fullname" . }}"] + verbs: + - get +``` +{{- end }} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl new file mode 100644 index 000000000..6f6518b71 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl @@ -0,0 +1,202 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "prometheus-node-exporter.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 "prometheus-node-exporter.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 "prometheus-node-exporter.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "prometheus-node-exporter.labels" -}} +helm.sh/chart: {{ include "prometheus-node-exporter.chart" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/component: metrics +app.kubernetes.io/part-of: {{ include "prometheus-node-exporter.name" . }} +{{ include "prometheus-node-exporter.selectorLabels" . }} +{{- with .Chart.AppVersion }} +app.kubernetes.io/version: {{ . | quote }} +{{- end }} +{{- with .Values.commonLabels }} +{{ tpl (toYaml .) $ }} +{{- end }} +{{- if .Values.releaseLabel }} +release: {{ .Release.Name }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "prometheus-node-exporter.selectorLabels" -}} +app.kubernetes.io/name: {{ include "prometheus-node-exporter.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + + +{{/* +Create the name of the service account to use +*/}} +{{- define "prometheus-node-exporter.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "prometheus-node-exporter.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +The image to use +*/}} +{{- define "prometheus-node-exporter.image" -}} +{{- if .Values.image.sha }} +{{- fail "image.sha forbidden. Use image.digest instead" }} +{{- else if .Values.image.digest }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s@%s" .Values.global.imageRegistry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) .Values.image.digest }} +{{- else }} +{{- printf "%s/%s:%s@%s" .Values.image.registry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) .Values.image.digest }} +{{- end }} +{{- else }} +{{- if .Values.global.imageRegistry }} +{{- printf "%s/%s:%s" .Values.global.imageRegistry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) }} +{{- else }} +{{- printf "%s/%s:%s" .Values.image.registry .Values.image.repository (default (printf "v%s" .Chart.AppVersion) .Values.image.tag) }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts +*/}} +{{- define "prometheus-node-exporter.namespace" -}} +{{- if .Values.namespaceOverride }} +{{- .Values.namespaceOverride }} +{{- else }} +{{- .Release.Namespace }} +{{- end }} +{{- end }} + +{{/* +Create the namespace name of the service monitor +*/}} +{{- define "prometheus-node-exporter.monitor-namespace" -}} +{{- if .Values.namespaceOverride }} +{{- .Values.namespaceOverride }} +{{- else }} +{{- if .Values.prometheus.monitor.namespace }} +{{- .Values.prometheus.monitor.namespace }} +{{- else }} +{{- .Release.Namespace }} +{{- end }} +{{- end }} +{{- end }} + +{{/* Sets default scrape limits for servicemonitor */}} +{{- define "servicemonitor.scrapeLimits" -}} +{{- with .sampleLimit }} +sampleLimit: {{ . }} +{{- end }} +{{- with .targetLimit }} +targetLimit: {{ . }} +{{- end }} +{{- with .labelLimit }} +labelLimit: {{ . }} +{{- end }} +{{- with .labelNameLengthLimit }} +labelNameLengthLimit: {{ . }} +{{- end }} +{{- with .labelValueLengthLimit }} +labelValueLengthLimit: {{ . }} +{{- end }} +{{- end }} + +{{/* +Formats imagePullSecrets. Input is (dict "Values" .Values "imagePullSecrets" .{specific imagePullSecrets}) +*/}} +{{- define "prometheus-node-exporter.imagePullSecrets" -}} +{{- range (concat .Values.global.imagePullSecrets .imagePullSecrets) }} + {{- if eq (typeOf .) "map[string]interface {}" }} +- {{ toYaml . | trim }} + {{- else }} +- name: {{ . }} + {{- end }} +{{- end }} +{{- end -}} + +{{/* +Create the namespace name of the pod monitor +*/}} +{{- define "prometheus-node-exporter.podmonitor-namespace" -}} +{{- if .Values.namespaceOverride }} +{{- .Values.namespaceOverride }} +{{- else }} +{{- if .Values.prometheus.podMonitor.namespace }} +{{- .Values.prometheus.podMonitor.namespace }} +{{- else }} +{{- .Release.Namespace }} +{{- end }} +{{- end }} +{{- end }} + +{{/* Sets default scrape limits for podmonitor */}} +{{- define "podmonitor.scrapeLimits" -}} +{{- with .sampleLimit }} +sampleLimit: {{ . }} +{{- end }} +{{- with .targetLimit }} +targetLimit: {{ . }} +{{- end }} +{{- with .labelLimit }} +labelLimit: {{ . }} +{{- end }} +{{- with .labelNameLengthLimit }} +labelNameLengthLimit: {{ . }} +{{- end }} +{{- with .labelValueLengthLimit }} +labelValueLengthLimit: {{ . }} +{{- end }} +{{- end }} + +{{/* Sets sidecar volumeMounts */}} +{{- define "prometheus-node-exporter.sidecarVolumeMounts" -}} +{{- range $_, $mount := $.Values.sidecarVolumeMount }} +- name: {{ $mount.name }} + mountPath: {{ $mount.mountPath }} + readOnly: {{ $mount.readOnly }} +{{- end }} +{{- range $_, $mount := $.Values.sidecarHostVolumeMounts }} +- name: {{ $mount.name }} + mountPath: {{ $mount.mountPath }} + readOnly: {{ $mount.readOnly }} +{{- if $mount.mountPropagation }} + mountPropagation: {{ $mount.mountPropagation }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml new file mode 100644 index 000000000..c256dba73 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml @@ -0,0 +1,19 @@ +{{- if and (eq .Values.rbac.create true) (eq .Values.kubeRBACProxy.enabled true) -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} +rules: + {{- if $.Values.kubeRBACProxy.enabled }} + - apiGroups: [ "authentication.k8s.io" ] + resources: + - tokenreviews + verbs: [ "create" ] + - apiGroups: [ "authorization.k8s.io" ] + resources: + - subjectaccessreviews + verbs: [ "create" ] + {{- end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..653305ad9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and (eq .Values.rbac.create true) (eq .Values.kubeRBACProxy.enabled true) -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + name: {{ template "prometheus-node-exporter.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole +{{- if .Values.rbac.useExistingRole }} + name: {{ .Values.rbac.useExistingRole }} +{{- else }} + name: {{ template "prometheus-node-exporter.fullname" . }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "prometheus-node-exporter.serviceAccountName" . }} + namespace: {{ template "prometheus-node-exporter.namespace" . }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml new file mode 100644 index 000000000..37ac60e67 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml @@ -0,0 +1,315 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + {{- with .Values.daemonsetAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 6 }} + revisionHistoryLimit: {{ .Values.revisionHistoryLimit }} + {{- with .Values.updateStrategy }} + updateStrategy: + {{- toYaml . | nindent 4 }} + {{- end }} + template: + metadata: + {{- with .Values.podAnnotations }} + annotations: + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- tpl (toYaml .) $ | nindent 8 }} + {{- end }} + spec: + automountServiceAccountToken: {{ ternary true false (or .Values.serviceAccount.automountServiceAccountToken .Values.kubeRBACProxy.enabled) }} + {{- with .Values.securityContext }} + securityContext: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.priorityClassName }} + priorityClassName: {{ . }} + {{- end }} + {{- with .Values.extraInitContainers }} + initContainers: + {{- toYaml . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "prometheus-node-exporter.serviceAccountName" . }} + {{- with .Values.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ . }} + {{- end }} + containers: + {{- $servicePort := ternary .Values.kubeRBACProxy.port .Values.service.port .Values.kubeRBACProxy.enabled }} + - name: node-exporter + image: {{ include "prometheus-node-exporter.image" . }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + args: + - --path.procfs=/host/proc + - --path.sysfs=/host/sys + {{- if .Values.hostRootFsMount.enabled }} + - --path.rootfs=/host/root + {{- if semverCompare ">=1.4.0-0" (coalesce .Values.version .Values.image.tag .Chart.AppVersion) }} + - --path.udev.data=/host/root/run/udev/data + {{- end }} + {{- end }} + - --web.listen-address=[$(HOST_IP)]:{{ $servicePort }} + {{- with .Values.extraArgs }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + env: + - name: HOST_IP + {{- if .Values.kubeRBACProxy.enabled }} + value: 127.0.0.1 + {{- else if .Values.service.listenOnAllInterfaces }} + value: 0.0.0.0 + {{- else }} + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: status.hostIP + {{- end }} + {{- range $key, $value := .Values.env }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- if eq .Values.kubeRBACProxy.enabled false }} + ports: + - name: {{ .Values.service.portName }} + containerPort: {{ .Values.service.port }} + protocol: TCP + {{- end }} + livenessProbe: + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + httpGet: + {{- if .Values.kubeRBACProxy.enabled }} + host: 127.0.0.1 + {{- end }} + httpHeaders: + {{- range $_, $header := .Values.livenessProbe.httpGet.httpHeaders }} + - name: {{ $header.name }} + value: {{ $header.value }} + {{- end }} + path: / + port: {{ $servicePort }} + scheme: {{ upper .Values.livenessProbe.httpGet.scheme }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + readinessProbe: + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + httpGet: + {{- if .Values.kubeRBACProxy.enabled }} + host: 127.0.0.1 + {{- end }} + httpHeaders: + {{- range $_, $header := .Values.readinessProbe.httpGet.httpHeaders }} + - name: {{ $header.name }} + value: {{ $header.value }} + {{- end }} + path: / + port: {{ $servicePort }} + scheme: {{ upper .Values.readinessProbe.httpGet.scheme }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.terminationMessageParams.enabled }} + {{- with .Values.terminationMessageParams }} + terminationMessagePath: {{ .terminationMessagePath }} + terminationMessagePolicy: {{ .terminationMessagePolicy }} + {{- end }} + {{- end }} + volumeMounts: + - name: proc + mountPath: /host/proc + {{- with .Values.hostProcFsMount.mountPropagation }} + mountPropagation: {{ . }} + {{- end }} + readOnly: true + - name: sys + mountPath: /host/sys + {{- with .Values.hostSysFsMount.mountPropagation }} + mountPropagation: {{ . }} + {{- end }} + readOnly: true + {{- if .Values.hostRootFsMount.enabled }} + - name: root + mountPath: /host/root + {{- with .Values.hostRootFsMount.mountPropagation }} + mountPropagation: {{ . }} + {{- end }} + readOnly: true + {{- end }} + {{- range $_, $mount := .Values.extraHostVolumeMounts }} + - name: {{ $mount.name }} + mountPath: {{ $mount.mountPath }} + readOnly: {{ $mount.readOnly }} + {{- with $mount.mountPropagation }} + mountPropagation: {{ . }} + {{- end }} + {{- end }} + {{- range $_, $mount := .Values.sidecarVolumeMount }} + - name: {{ $mount.name }} + mountPath: {{ $mount.mountPath }} + readOnly: true + {{- end }} + {{- range $_, $mount := .Values.configmaps }} + - name: {{ $mount.name }} + mountPath: {{ $mount.mountPath }} + {{- end }} + {{- range $_, $mount := .Values.secrets }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + {{- end }} + {{- range .Values.sidecars }} + {{- $overwrites := dict "volumeMounts" (concat (include "prometheus-node-exporter.sidecarVolumeMounts" $ | fromYamlArray) (.volumeMounts | default list) | default list) }} + {{- $defaults := dict "image" (include "prometheus-node-exporter.image" $) "securityContext" $.Values.containerSecurityContext "imagePullPolicy" $.Values.image.pullPolicy }} + - {{- toYaml (merge $overwrites . $defaults) | nindent 10 }} + {{- end }} + {{- if .Values.kubeRBACProxy.enabled }} + - name: kube-rbac-proxy + args: + {{- if .Values.kubeRBACProxy.extraArgs }} + {{- .Values.kubeRBACProxy.extraArgs | toYaml | nindent 12 }} + {{- end }} + - --secure-listen-address=:{{ .Values.service.port}} + - --upstream=http://127.0.0.1:{{ $servicePort }}/ + - --proxy-endpoints-port={{ .Values.kubeRBACProxy.proxyEndpointsPort }} + - --config-file=/etc/kube-rbac-proxy-config/config-file.yaml + volumeMounts: + - name: kube-rbac-proxy-config + mountPath: /etc/kube-rbac-proxy-config + imagePullPolicy: {{ .Values.kubeRBACProxy.image.pullPolicy }} + {{- if .Values.kubeRBACProxy.image.sha }} + image: "{{ .Values.global.imageRegistry | default .Values.kubeRBACProxy.image.registry}}/{{ .Values.kubeRBACProxy.image.repository }}:{{ .Values.kubeRBACProxy.image.tag }}@sha256:{{ .Values.kubeRBACProxy.image.sha }}" + {{- else }} + image: "{{ .Values.global.imageRegistry | default .Values.kubeRBACProxy.image.registry}}/{{ .Values.kubeRBACProxy.image.repository }}:{{ .Values.kubeRBACProxy.image.tag }}" + {{- end }} + ports: + - containerPort: {{ .Values.service.port}} + name: {{ .Values.kubeRBACProxy.portName }} + {{- if .Values.kubeRBACProxy.enableHostPort }} + hostPort: {{ .Values.service.port }} + {{- end }} + - containerPort: {{ .Values.kubeRBACProxy.proxyEndpointsPort }} + {{- if .Values.kubeRBACProxy.enableProxyEndpointsHostPort }} + hostPort: {{ .Values.kubeRBACProxy.proxyEndpointsPort }} + {{- end }} + name: "http-healthz" + readinessProbe: + httpGet: + scheme: HTTPS + port: {{ .Values.kubeRBACProxy.proxyEndpointsPort }} + path: healthz + initialDelaySeconds: 5 + timeoutSeconds: 5 + {{- if .Values.kubeRBACProxy.resources }} + resources: + {{- toYaml .Values.kubeRBACProxy.resources | nindent 12 }} + {{- end }} + {{- if .Values.terminationMessageParams.enabled }} + {{- with .Values.terminationMessageParams }} + terminationMessagePath: {{ .terminationMessagePath }} + terminationMessagePolicy: {{ .terminationMessagePolicy }} + {{- end }} + {{- end }} + {{- with .Values.kubeRBACProxy.env }} + env: + {{- range $key, $value := $.Values.kubeRBACProxy.env }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- end }} + {{- if .Values.kubeRBACProxy.containerSecurityContext }} + securityContext: + {{ toYaml .Values.kubeRBACProxy.containerSecurityContext | nindent 12 }} + {{- end }} + {{- end }} + {{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} + imagePullSecrets: + {{- include "prometheus-node-exporter.imagePullSecrets" (dict "Values" .Values "imagePullSecrets" .Values.imagePullSecrets) | indent 8 }} + {{- end }} + hostNetwork: {{ .Values.hostNetwork }} + hostPID: {{ .Values.hostPID }} + hostIPC: {{ .Values.hostIPC }} + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.dnsConfig }} + dnsConfig: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.restartPolicy }} + restartPolicy: {{ . }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: proc + hostPath: + path: /proc + - name: sys + hostPath: + path: /sys + {{- if .Values.hostRootFsMount.enabled }} + - name: root + hostPath: + path: / + {{- end }} + {{- range $_, $mount := .Values.extraHostVolumeMounts }} + - name: {{ $mount.name }} + hostPath: + path: {{ $mount.hostPath }} + {{- with $mount.type }} + type: {{ . }} + {{- end }} + {{- end }} + {{- range $_, $mount := .Values.sidecarVolumeMount }} + - name: {{ $mount.name }} + emptyDir: + medium: Memory + {{- end }} + {{- range $_, $mount := .Values.sidecarHostVolumeMounts }} + - name: {{ $mount.name }} + hostPath: + path: {{ $mount.hostPath }} + {{- end }} + {{- range $_, $mount := .Values.configmaps }} + - name: {{ $mount.name }} + configMap: + name: {{ $mount.name }} + {{- end }} + {{- range $_, $mount := .Values.secrets }} + - name: {{ $mount.name }} + secret: + secretName: {{ $mount.name }} + {{- end }} + {{- if .Values.kubeRBACProxy.enabled }} + - name: kube-rbac-proxy-config + configMap: + name: {{ template "prometheus-node-exporter.fullname" . }}-rbac-config + {{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml new file mode 100644 index 000000000..45eeb8d96 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml @@ -0,0 +1,18 @@ +{{- if .Values.endpoints }} +apiVersion: v1 +kind: Endpoints +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} +subsets: + - addresses: + {{- range .Values.endpoints }} + - ip: {{ . }} + {{- end }} + ports: + - name: {{ .Values.service.portName }} + port: 9100 + protocol: TCP +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml new file mode 100644 index 000000000..2b21b7106 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraManifests }} +--- +{{ tpl . $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml new file mode 100644 index 000000000..ee4090210 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml @@ -0,0 +1,27 @@ +{{- if .Values.networkPolicy.enabled }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" $ | nindent 4 }} + {{- with .Values.service.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ingress: + {{- if .Values.networkPolicy.ingress }} + {{- toYaml .Values.networkPolicy.ingress | nindent 4 }} + {{- else }} + - ports: + - port: {{ .Values.service.port }} + {{- end }} + policyTypes: + - Egress + - Ingress + podSelector: + matchLabels: + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml new file mode 100644 index 000000000..f88da6a34 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml @@ -0,0 +1,91 @@ +{{- if .Values.prometheus.podMonitor.enabled }} +apiVersion: {{ .Values.prometheus.podMonitor.apiVersion | default "monitoring.coreos.com/v1" }} +kind: PodMonitor +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.podmonitor-namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + {{- with .Values.prometheus.podMonitor.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + jobLabel: {{ default "app.kubernetes.io/name" .Values.prometheus.podMonitor.jobLabel }} + {{- include "podmonitor.scrapeLimits" .Values.prometheus.podMonitor | nindent 2 }} + selector: + matchLabels: + {{- with .Values.prometheus.podMonitor.selectorOverride }} + {{- toYaml . | nindent 6 }} + {{- else }} + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 6 }} + {{- end }} + namespaceSelector: + matchNames: + - {{ include "prometheus-node-exporter.namespace" . }} + {{- with .Values.prometheus.podMonitor.attachMetadata }} + attachMetadata: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.podTargetLabels }} + podTargetLabels: + {{- toYaml . | nindent 4 }} + {{- end }} + podMetricsEndpoints: + - port: {{ .Values.service.portName }} + {{- with .Values.prometheus.podMonitor.scheme }} + scheme: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.path }} + path: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.basicAuth }} + basicAuth: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.bearerTokenSecret }} + bearerTokenSecret: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.authorization }} + authorization: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.oauth2 }} + oauth2: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.proxyUrl }} + proxyUrl: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.honorTimestamps }} + honorTimestamps: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.honorLabels }} + honorLabels: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + {{- with .Values.prometheus.podMonitor.relabelings }} + relabelings: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.podMonitor.metricRelabelings }} + metricRelabelings: + {{- toYaml . | nindent 8 }} + {{- end }} + enableHttp2: {{ default false .Values.prometheus.podMonitor.enableHttp2 }} + filterRunning: {{ default true .Values.prometheus.podMonitor.filterRunning }} + followRedirects: {{ default false .Values.prometheus.podMonitor.followRedirects }} + {{- with .Values.prometheus.podMonitor.params }} + params: + {{- toYaml . | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml new file mode 100644 index 000000000..895731724 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml @@ -0,0 +1,14 @@ +{{- if and .Values.rbac.create .Values.rbac.pspEnabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: psp-{{ include "prometheus-node-exporter.fullname" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} +rules: +- apiGroups: ['extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ include "prometheus-node-exporter.fullname" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml new file mode 100644 index 000000000..333370173 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.rbac.create .Values.rbac.pspEnabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: psp-{{ include "prometheus-node-exporter.fullname" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp-{{ include "prometheus-node-exporter.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml new file mode 100644 index 000000000..4896c84da --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml @@ -0,0 +1,49 @@ +{{- if and .Values.rbac.create .Values.rbac.pspEnabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + {{- with .Values.rbac.pspAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + privileged: false + # Allow core volume types. + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + - 'hostPath' + hostNetwork: true + hostIPC: false + hostPID: true + hostPorts: + - min: 0 + max: 65535 + runAsUser: + # Permits the container to run with root privileges as well. + rule: 'RunAsAny' + seLinux: + # This policy assumes the nodes are using AppArmor rather than SELinux. + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Allow adding the root group. + - min: 0 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Allow adding the root group. + - min: 0 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml new file mode 100644 index 000000000..814e11033 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml @@ -0,0 +1,16 @@ +{{- if .Values.kubeRBACProxy.enabled}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "prometheus-node-exporter.fullname" . }}-rbac-config + namespace: {{ include "prometheus-node-exporter.namespace" . }} +data: + config-file.yaml: |+ + authorization: + resourceAttributes: + namespace: {{ template "prometheus-node-exporter.namespace" . }} + apiVersion: v1 + resource: services + subresource: {{ template "prometheus-node-exporter.fullname" . }} + name: {{ template "prometheus-node-exporter.fullname" . }} +{{- end }} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml new file mode 100644 index 000000000..abaa31b7f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml @@ -0,0 +1,38 @@ +{{- if .Values.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" $ | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.service.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: +{{- if .Values.service.ipDualStack.enabled }} + ipFamilies: {{ toYaml .Values.service.ipDualStack.ipFamilies | nindent 4 }} + ipFamilyPolicy: {{ .Values.service.ipDualStack.ipFamilyPolicy }} +{{- end }} +{{- if .Values.service.externalTrafficPolicy }} + externalTrafficPolicy: {{ .Values.service.externalTrafficPolicy }} +{{- end }} + type: {{ .Values.service.type }} +{{- if and (eq .Values.service.type "ClusterIP") .Values.service.clusterIP }} + clusterIP: "{{ .Values.service.clusterIP }}" +{{- end }} + ports: + - port: {{ .Values.service.servicePort | default .Values.service.port }} + {{- if ( and (eq .Values.service.type "NodePort" ) (not (empty .Values.service.nodePort)) ) }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + targetPort: {{ .Values.service.targetPort }} + protocol: TCP + name: {{ .Values.service.portName }} + selector: + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml new file mode 100644 index 000000000..462b0cda4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml @@ -0,0 +1,18 @@ +{{- if and .Values.rbac.create .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "prometheus-node-exporter.serviceAccountName" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} +{{- if or .Values.serviceAccount.imagePullSecrets .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- include "prometheus-node-exporter.imagePullSecrets" (dict "Values" .Values "imagePullSecrets" .Values.serviceAccount.imagePullSecrets) | indent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml new file mode 100644 index 000000000..0d7a42eae --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml @@ -0,0 +1,61 @@ +{{- if .Values.prometheus.monitor.enabled }} +apiVersion: {{ .Values.prometheus.monitor.apiVersion | default "monitoring.coreos.com/v1" }} +kind: ServiceMonitor +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.monitor-namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} + {{- with .Values.prometheus.monitor.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + jobLabel: {{ default "app.kubernetes.io/name" .Values.prometheus.monitor.jobLabel }} + {{- include "servicemonitor.scrapeLimits" .Values.prometheus.monitor | nindent 2 }} + {{- with .Values.prometheus.monitor.podTargetLabels }} + podTargetLabels: + {{- toYaml . | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- with .Values.prometheus.monitor.selectorOverride }} + {{- toYaml . | nindent 6 }} + {{- else }} + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 6 }} + {{- end }} + {{- with .Values.prometheus.monitor.attachMetadata }} + attachMetadata: + {{- toYaml . | nindent 4 }} + {{- end }} + endpoints: + - port: {{ .Values.service.portName }} + scheme: {{ .Values.prometheus.monitor.scheme }} + {{- with .Values.prometheus.monitor.basicAuth }} + basicAuth: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.monitor.bearerTokenFile }} + bearerTokenFile: {{ . }} + {{- end }} + {{- with .Values.prometheus.monitor.tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.monitor.proxyUrl }} + proxyUrl: {{ . }} + {{- end }} + {{- with .Values.prometheus.monitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.prometheus.monitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + {{- with .Values.prometheus.monitor.relabelings }} + relabelings: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.prometheus.monitor.metricRelabelings }} + metricRelabelings: + {{- toYaml . | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml new file mode 100644 index 000000000..2c2705f87 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml @@ -0,0 +1,40 @@ +{{- if and (.Capabilities.APIVersions.Has "autoscaling.k8s.io/v1") (.Values.verticalPodAutoscaler.enabled) }} +apiVersion: autoscaling.k8s.io/v1 +kind: VerticalPodAutoscaler +metadata: + name: {{ include "prometheus-node-exporter.fullname" . }} + namespace: {{ include "prometheus-node-exporter.namespace" . }} + labels: + {{- include "prometheus-node-exporter.labels" . | nindent 4 }} +spec: + {{- with .Values.verticalPodAutoscaler.recommenders }} + recommenders: + {{- toYaml . | nindent 4 }} + {{- end }} + resourcePolicy: + containerPolicies: + - containerName: node-exporter + {{- with .Values.verticalPodAutoscaler.controlledResources }} + controlledResources: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.verticalPodAutoscaler.controlledValues }} + controlledValues: {{ . }} + {{- end }} + {{- with .Values.verticalPodAutoscaler.maxAllowed }} + maxAllowed: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.verticalPodAutoscaler.minAllowed }} + minAllowed: + {{- toYaml . | nindent 8 }} + {{- end }} + targetRef: + apiVersion: apps/v1 + kind: DaemonSet + name: {{ include "prometheus-node-exporter.fullname" . }} + {{- with .Values.verticalPodAutoscaler.updatePolicy }} + updatePolicy: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/values.yaml new file mode 100644 index 000000000..73a8f4a2e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-node-exporter/values.yaml @@ -0,0 +1,566 @@ +# Default values for prometheus-node-exporter. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. +image: + registry: quay.io + repository: prometheus/node-exporter + # Overrides the image tag whose default is {{ printf "v%s" .Chart.AppVersion }} + tag: "" + pullPolicy: IfNotPresent + digest: "" + +imagePullSecrets: [] +# - name: "image-pull-secret" +nameOverride: "" +fullnameOverride: "" + +# Number of old history to retain to allow rollback +# Default Kubernetes value is set to 10 +revisionHistoryLimit: 10 + +global: + # To help compatibility with other charts which use global.imagePullSecrets. + # Allow either an array of {name: pullSecret} maps (k8s-style), or an array of strings (more common helm-style). + # global: + # imagePullSecrets: + # - name: pullSecret1 + # - name: pullSecret2 + # or + # global: + # imagePullSecrets: + # - pullSecret1 + # - pullSecret2 + imagePullSecrets: [] + # + # Allow parent charts to override registry hostname + imageRegistry: "" + +# Configure kube-rbac-proxy. When enabled, creates a kube-rbac-proxy to protect the node-exporter http endpoint. +# The requests are served through the same service but requests are HTTPS. +kubeRBACProxy: + enabled: false + ## Set environment variables as name/value pairs + env: {} + # VARIABLE: value + image: + registry: quay.io + repository: brancz/kube-rbac-proxy + tag: v0.18.0 + sha: "" + pullPolicy: IfNotPresent + + # List of additional cli arguments to configure kube-rbac-proxy + # for example: --tls-cipher-suites, --log-file, etc. + # all the possible args can be found here: https://github.com/brancz/kube-rbac-proxy#usage + extraArgs: [] + + ## Specify security settings for a Container + ## Allows overrides and additional options compared to (Pod) securityContext + ## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container + containerSecurityContext: {} + + # Specify the port used for the Node exporter container (upstream port) + port: 8100 + # Specify the name of the container port + portName: http + # Configure a hostPort. If true, hostPort will be enabled in the container and set to service.port. + enableHostPort: false + + # Configure Proxy Endpoints Port + # This is the port being probed for readiness + proxyEndpointsPort: 8888 + # Configure a hostPort. If true, hostPort will be enabled in the container and set to proxyEndpointsPort. + enableProxyEndpointsHostPort: false + + 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: 64Mi + # requests: + # cpu: 10m + # memory: 32Mi + +## Service configuration +service: + ## Creating a service is enabled by default + enabled: true + + ## Service type + type: ClusterIP + ## IP address for type ClusterIP + clusterIP: "" + ## Default service port. Sets the port of the exposed container as well (NE or kubeRBACProxy). + ## Use "servicePort" below if changing the service port only is desired. + port: 9100 + ## Service port. Use this field if you wish to set a different service port + ## without changing the container port ("port" above). + servicePort: "" + ## Targeted port in the pod. Must refer to an open container port ("port" or "portName"). + ## (IntOrString) + targetPort: 9100 + ## Name of the service port. Sets the port name of the main container (NE) as well. + portName: metrics + ## Port number for service type NodePort + nodePort: null + + ## If true, node exporter will listen on all interfaces + listenOnAllInterfaces: true + + ## Additional annotations and labels for the service + annotations: + prometheus.io/scrape: "true" + labels: {} + + ## Dual stack settings for the service + ## https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ipDualStack: + enabled: false + ipFamilies: ["IPv6", "IPv4"] + ipFamilyPolicy: "PreferDualStack" + + ## External traffic policy setting (Cluster, Local) + externalTrafficPolicy: "" + +# Set a NetworkPolicy with: +# ingress only on service.port or custom policy +# no egress permitted +networkPolicy: + enabled: false + + # ingress: + # - {} + +# Additional environment variables that will be passed to the daemonset +env: {} +## env: +## VARIABLE: value + +prometheus: + monitor: + enabled: false + additionalLabels: {} + namespace: "" + + jobLabel: "" + + # List of pod labels to add to node exporter metrics + # https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#servicemonitor + podTargetLabels: [] + + scheme: http + basicAuth: {} + bearerTokenFile: + tlsConfig: {} + + ## proxyUrl: URL of a proxy that should be used for scraping. + ## + proxyUrl: "" + + ## Override serviceMonitor selector + ## + selectorOverride: {} + + ## Attach node metadata to discovered targets. Requires Prometheus v2.35.0 and above. + ## + attachMetadata: + node: false + + relabelings: [] + metricRelabelings: [] + interval: "" + scrapeTimeout: 10s + ## prometheus.monitor.apiVersion ApiVersion for the serviceMonitor Resource(defaults to "monitoring.coreos.com/v1") + apiVersion: "" + + ## SampleLimit defines per-scrape limit on number of scraped samples that will be accepted. + ## + sampleLimit: 0 + + ## TargetLimit defines a limit on the number of scraped targets that will be accepted. + ## + targetLimit: 0 + + ## Per-scrape limit on number of labels that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelLimit: 0 + + ## Per-scrape limit on length of labels name that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelNameLengthLimit: 0 + + ## Per-scrape limit on length of labels value that will be accepted for a sample. Only valid in Prometheus versions 2.27.0 and newer. + ## + labelValueLengthLimit: 0 + + # PodMonitor defines monitoring for a set of pods. + # ref. https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#monitoring.coreos.com/v1.PodMonitor + # Using a PodMonitor may be preferred in some environments where there is very large number + # of Node Exporter endpoints (1000+) behind a single service. + # The PodMonitor is disabled by default. When switching from ServiceMonitor to PodMonitor, + # the time series resulting from the configuration through PodMonitor may have different labels. + # For instance, there will not be the service label any longer which might + # affect PromQL queries selecting that label. + podMonitor: + enabled: false + # Namespace in which to deploy the pod monitor. Defaults to the release namespace. + namespace: "" + # Additional labels, e.g. setting a label for pod monitor selector as set in prometheus + additionalLabels: {} + # release: kube-prometheus-stack + # PodTargetLabels transfers labels of the Kubernetes Pod onto the target. + podTargetLabels: [] + # apiVersion defaults to monitoring.coreos.com/v1. + apiVersion: "" + # Override pod selector to select pod objects. + selectorOverride: {} + # Attach node metadata to discovered targets. Requires Prometheus v2.35.0 and above. + attachMetadata: + node: false + # The label to use to retrieve the job name from. Defaults to label app.kubernetes.io/name. + jobLabel: "" + + # Scheme/protocol to use for scraping. + scheme: "http" + # Path to scrape metrics at. + path: "/metrics" + + # BasicAuth allow an endpoint to authenticate over basic authentication. + # More info: https://prometheus.io/docs/operating/configuration/#endpoint + basicAuth: {} + # Secret to mount to read bearer token for scraping targets. + # The secret needs to be in the same namespace as the pod monitor and accessible by the Prometheus Operator. + # https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.24/#secretkeyselector-v1-core + bearerTokenSecret: {} + # TLS configuration to use when scraping the endpoint. + tlsConfig: {} + # Authorization section for this endpoint. + # https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#monitoring.coreos.com/v1.SafeAuthorization + authorization: {} + # OAuth2 for the URL. Only valid in Prometheus versions 2.27.0 and newer. + # https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#monitoring.coreos.com/v1.OAuth2 + oauth2: {} + + # ProxyURL eg http://proxyserver:2195. Directs scrapes through proxy to this endpoint. + proxyUrl: "" + # Interval at which endpoints should be scraped. If not specified Prometheus’ global scrape interval is used. + interval: "" + # Timeout after which the scrape is ended. If not specified, the Prometheus global scrape interval is used. + scrapeTimeout: "" + # HonorTimestamps controls whether Prometheus respects the timestamps present in scraped data. + honorTimestamps: true + # HonorLabels chooses the metric’s labels on collisions with target labels. + honorLabels: true + # Whether to enable HTTP2. Default false. + enableHttp2: "" + # Drop pods that are not running. (Failed, Succeeded). + # Enabled by default. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#pod-phase + filterRunning: "" + # FollowRedirects configures whether scrape requests follow HTTP 3xx redirects. Default false. + followRedirects: "" + # Optional HTTP URL parameters + params: {} + + # RelabelConfigs to apply to samples before scraping. Prometheus Operator automatically adds + # relabelings for a few standard Kubernetes fields. The original scrape job’s name + # is available via the __tmp_prometheus_job_name label. + # More info: https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config + relabelings: [] + # MetricRelabelConfigs to apply to samples before ingestion. + metricRelabelings: [] + + # SampleLimit defines per-scrape limit on number of scraped samples that will be accepted. + sampleLimit: 0 + # TargetLimit defines a limit on the number of scraped targets that will be accepted. + targetLimit: 0 + # Per-scrape limit on number of labels that will be accepted for a sample. + # Only valid in Prometheus versions 2.27.0 and newer. + labelLimit: 0 + # Per-scrape limit on length of labels name that will be accepted for a sample. + # Only valid in Prometheus versions 2.27.0 and newer. + labelNameLengthLimit: 0 + # Per-scrape limit on length of labels value that will be accepted for a sample. + # Only valid in Prometheus versions 2.27.0 and newer. + labelValueLengthLimit: 0 + +## Customize the updateStrategy if set +updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + +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: 200m + # memory: 50Mi + # requests: + # cpu: 100m + # memory: 30Mi + +# Specify the container restart policy passed to the Node Export container +# Possible Values: Always (default)|OnFailure|Never +restartPolicy: null + +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: + annotations: {} + imagePullSecrets: [] + automountServiceAccountToken: false + +securityContext: + fsGroup: 65534 + runAsGroup: 65534 + runAsNonRoot: true + runAsUser: 65534 + +containerSecurityContext: + readOnlyRootFilesystem: true + # capabilities: + # add: + # - SYS_TIME + +rbac: + ## If true, create & use RBAC resources + ## + create: true + ## If true, create & use Pod Security Policy resources + ## https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + pspEnabled: true + pspAnnotations: {} + +# for deployments that have node_exporter deployed outside of the cluster, list +# their addresses here +endpoints: [] + +# Expose the service to the host network +hostNetwork: true + +# Share the host process ID namespace +hostPID: true + +# Share the host ipc namespace +hostIPC: false + +# Mount the node's root file system (/) at /host/root in the container +hostRootFsMount: + enabled: true + # Defines how new mounts in existing mounts on the node or in the container + # are propagated to the container or node, respectively. Possible values are + # None, HostToContainer, and Bidirectional. If this field is omitted, then + # None is used. More information on: + # https://kubernetes.io/docs/concepts/storage/volumes/#mount-propagation + mountPropagation: HostToContainer + +# Mount the node's proc file system (/proc) at /host/proc in the container +hostProcFsMount: + # Possible values are None, HostToContainer, and Bidirectional + mountPropagation: "" + +# Mount the node's sys file system (/sys) at /host/sys in the container +hostSysFsMount: + # Possible values are None, HostToContainer, and Bidirectional + mountPropagation: "" + +## Assign a group of affinity scheduling rules +## +affinity: {} +# nodeAffinity: +# requiredDuringSchedulingIgnoredDuringExecution: +# nodeSelectorTerms: +# - matchFields: +# - key: metadata.name +# operator: In +# values: +# - target-host-name + +# Annotations to be added to node exporter pods +podAnnotations: + # Fix for very slow GKE cluster upgrades + cluster-autoscaler.kubernetes.io/safe-to-evict: "true" + +# Extra labels to add to node exporter pods (can be templated) +podLabels: {} + +## Extra labels to attach to all resources (can be templated) +commonLabels: {} + +# Annotations to be added to node exporter daemonset +daemonsetAnnotations: {} + +## set to true to add the release label so scraping of the servicemonitor with kube-prometheus-stack works out of the box +releaseLabel: false + +# Custom DNS configuration to be added to prometheus-node-exporter pods +dnsConfig: {} +# nameservers: +# - 1.2.3.4 +# searches: +# - ns1.svc.cluster-domain.example +# - my.dns.search.suffix +# options: +# - name: ndots +# value: "2" +# - name: edns0 + +## Assign a nodeSelector if operating a hybrid cluster +## +nodeSelector: + kubernetes.io/os: linux + # kubernetes.io/arch: amd64 + +# Specify grace period for graceful termination of pods. Defaults to 30 if null or not specified +terminationGracePeriodSeconds: null + +tolerations: + - effect: NoSchedule + operator: Exists + +# Enable or disable container termination message settings +# https://kubernetes.io/docs/tasks/debug/debug-application/determine-reason-pod-failure/ +terminationMessageParams: + enabled: false + # If enabled, specify the path for termination messages + terminationMessagePath: /dev/termination-log + # If enabled, specify the policy for termination messages + terminationMessagePolicy: File + + +## Assign a PriorityClassName to pods if set +# priorityClassName: "" + +## Additional container arguments +## +extraArgs: [] +# - --collector.diskstats.ignored-devices=^(ram|loop|fd|(h|s|v)d[a-z]|nvme\\d+n\\d+p)\\d+$ +# - --collector.textfile.directory=/run/prometheus + +## Additional mounts from the host to node-exporter container +## +extraHostVolumeMounts: [] +# - name: +# hostPath: +# https://kubernetes.io/docs/concepts/storage/volumes/#hostpath-volume-types +# type: "" (Default)|DirectoryOrCreate|Directory|FileOrCreate|File|Socket|CharDevice|BlockDevice +# mountPath: +# readOnly: true|false +# mountPropagation: None|HostToContainer|Bidirectional + +## Additional configmaps to be mounted. +## +configmaps: [] +# - name: +# mountPath: +secrets: [] +# - name: +# mountPath: +## Override the deployment namespace +## +namespaceOverride: "" + +## Additional containers for export metrics to text file; fields image,imagePullPolicy,securityContext take default value from main container +## +sidecars: [] +# - name: nvidia-dcgm-exporter +# image: nvidia/dcgm-exporter:1.4.3 +# volumeMounts: +# - name: tmp +# mountPath: /tmp + +## Volume for sidecar containers +## +sidecarVolumeMount: [] +# - name: collector-textfiles +# mountPath: /run/prometheus +# readOnly: false + +## Additional mounts from the host to sidecar containers +## +sidecarHostVolumeMounts: [] +# - name: +# hostPath: +# mountPath: +# readOnly: true|false +# mountPropagation: None|HostToContainer|Bidirectional + +## Additional InitContainers to initialize the pod +## +extraInitContainers: [] + +## Liveness probe +## +livenessProbe: + failureThreshold: 3 + httpGet: + httpHeaders: [] + scheme: http + initialDelaySeconds: 0 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + +## Readiness probe +## +readinessProbe: + failureThreshold: 3 + httpGet: + httpHeaders: [] + scheme: http + initialDelaySeconds: 0 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + +# Enable vertical pod autoscaler support for prometheus-node-exporter +verticalPodAutoscaler: + enabled: false + + # Recommender responsible for generating recommendation for the object. + # List should be empty (then the default recommender will generate the recommendation) + # or contain exactly one recommender. + # recommenders: + # - name: custom-recommender-performance + + # List of resources that the vertical pod autoscaler can control. Defaults to cpu and memory + controlledResources: [] + # Specifies which resource values should be controlled: RequestsOnly or RequestsAndLimits. + # controlledValues: RequestsAndLimits + + # Define the max allowed resources for the pod + maxAllowed: {} + # cpu: 200m + # memory: 100Mi + # Define the min allowed resources for the pod + minAllowed: {} + # cpu: 200m + # memory: 100Mi + + # updatePolicy: + # Specifies minimal number of replicas which need to be alive for VPA Updater to attempt pod eviction + # minReplicas: 1 + # Specifies whether recommended updates are applied when a Pod is started and whether recommended updates + # are applied during the life of a Pod. Possible values are "Off", "Initial", "Recreate", and "Auto". + # updateMode: Auto + +# Extra manifests to deploy as an array +extraManifests: [] + # - | + # apiVersion: v1 + # kind: ConfigMap + # metadata: + # name: prometheus-extra + # data: + # extra-data: "value" + +# Override version of app, required if image.tag is defined and does not follow semver +version: "" diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/.helmignore b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/.helmignore new file mode 100644 index 000000000..e90c9f6d2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/.helmignore @@ -0,0 +1,24 @@ +# 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 Kubernetes +OWNERS \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml new file mode 100644 index 000000000..501e8a69a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml @@ -0,0 +1,24 @@ +annotations: + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/prometheus-community/helm-charts +apiVersion: v2 +appVersion: v1.10.0 +description: A Helm chart for prometheus pushgateway +home: https://github.com/prometheus/pushgateway +keywords: +- pushgateway +- prometheus +maintainers: +- email: gianrubio@gmail.com + name: gianrubio +- email: christian.staude@staffbase.com + name: cstaud +- email: rootsandtrees@posteo.de + name: zeritti +name: prometheus-pushgateway +sources: +- https://github.com/prometheus/pushgateway +type: application +version: 2.15.0 diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/README.md b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/README.md new file mode 100644 index 000000000..cc6645fdf --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/README.md @@ -0,0 +1,88 @@ +# Prometheus Pushgateway + +This chart bootstraps a Prometheus [Pushgateway](http://github.com/prometheus/pushgateway) deployment on a [Kubernetes](http://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. + +An optional prometheus `ServiceMonitor` can be enabled, should you wish to use this gateway with [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +## Get Repository Info + +```console +helm repo add prometheus-community https://prometheus-community.github.io/helm-charts +helm repo update +``` + +_See [helm repo](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Install Chart + +```console +helm install [RELEASE_NAME] prometheus-community/prometheus-pushgateway +``` + +_See [configuration](#configuration) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Uninstall Chart + +```console +helm uninstall [RELEASE_NAME] +``` + +This removes all the Kubernetes components associated with the chart and deletes the release. + +_See [helm uninstall](https://helm.sh/docs/helm/helm_uninstall/) for command documentation._ + +## Upgrading Chart + +```console +helm upgrade [RELEASE_NAME] prometheus-community/prometheus-pushgateway --install +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +### To 2.0.0 + +Chart API version has been upgraded to v2 so Helm 3 is needed from now on. + +Docker image tag is used from Chart.yaml appVersion field by default now. + +Version 2.0.0 also adapted [Helm label and annotation best practices](https://helm.sh/docs/chart_best_practices/labels/). Specifically, labels mapping is listed below: + +```console +OLD => NEW +---------------------------------------- +heritage => app.kubernetes.io/managed-by +chart => helm.sh/chart +[container version] => app.kubernetes.io/version +app => app.kubernetes.io/name +release => app.kubernetes.io/instance +``` + +Therefore, depending on the way you've configured the chart, the previous StatefulSet or Deployment need to be deleted before upgrade. + +If `runAsStatefulSet: false` (this is the default): + +```console +kubectl delete deploy -l app=prometheus-pushgateway +``` + +If `runAsStatefulSet: true`: + +```console +kubectl delete sts -l app=prometheus-pushgateway +``` + +After that do the actual upgrade: + +```console +helm upgrade -i prometheus-pushgateway prometheus-community/prometheus-pushgateway +``` + +## Configuration + +See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). To see all configurable options with detailed comments, visit the chart's [values.yaml](./values.yaml), or run these configuration commands: + +```console +helm show values prometheus-community/prometheus-pushgateway +``` diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt new file mode 100644 index 000000000..263b1d8d4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt @@ -0,0 +1,19 @@ +1. Get the application URL by running these commands: +{{- if .Values.ingress.enabled }} +{{- range .Values.ingress.hosts }} + http{{ if $.Values.ingress.tls }}s{{ end }}://{{ . }}{{ $.Values.ingress.path }} +{{- end }} +{{- else if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ template "prometheus-pushgateway.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus-pushgateway.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ template "prometheus-pushgateway.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc -w {{ template "prometheus-pushgateway.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ template "prometheus-pushgateway.namespace" . }} {{ template "prometheus-pushgateway.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.service.port }} +{{- else if contains "ClusterIP" .Values.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ template "prometheus-pushgateway.namespace" . }} -l "app.kubernetes.io/name={{ template "prometheus-pushgateway.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl port-forward $POD_NAME 9091 + echo "Visit http://127.0.0.1:9091 to use your application" +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl new file mode 100644 index 000000000..dcd42ff36 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl @@ -0,0 +1,304 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "prometheus-pushgateway.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Namespace to set on the resources +*/}} +{{- define "prometheus-pushgateway.namespace" -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- end -}} +{{- 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 "prometheus-pushgateway.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 "prometheus-pushgateway.chart" -}} +{{ printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "prometheus-pushgateway.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "prometheus-pushgateway.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +Create default labels +*/}} +{{- define "prometheus-pushgateway.defaultLabels" -}} +helm.sh/chart: {{ include "prometheus-pushgateway.chart" . }} +{{ include "prometheus-pushgateway.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- with .Values.podLabels }} +{{ toYaml . }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "prometheus-pushgateway.selectorLabels" -}} +app.kubernetes.io/name: {{ include "prometheus-pushgateway.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Return the appropriate apiVersion for networkpolicy. +*/}} +{{- define "prometheus-pushgateway.networkPolicy.apiVersion" -}} +{{- if semverCompare ">=1.4-0, <1.7-0" .Capabilities.KubeVersion.GitVersion }} +{{- print "extensions/v1beta1" }} +{{- else if semverCompare "^1.7-0" .Capabilities.KubeVersion.GitVersion }} +{{- print "networking.k8s.io/v1" }} +{{- end }} +{{- end }} + +{{/* +Define PDB apiVersion +*/}} +{{- define "prometheus-pushgateway.pdb.apiVersion" -}} +{{- if $.Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} +{{- print "policy/v1" }} +{{- else }} +{{- print "policy/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Define Ingress apiVersion +*/}} +{{- define "prometheus-pushgateway.ingress.apiVersion" -}} +{{- if semverCompare ">=1.19-0" .Capabilities.KubeVersion.GitVersion }} +{{- print "networking.k8s.io/v1" }} +{{- else if semverCompare ">=1.14-0" .Capabilities.KubeVersion.GitVersion }} +{{- print "networking.k8s.io/v1beta1" }} +{{- else }} +{{- print "extensions/v1beta1" }} +{{- end }} +{{- end }} + +{{/* +Define webConfiguration +*/}} +{{- define "prometheus-pushgateway.webConfiguration" -}} +basic_auth_users: +{{- range $k, $v := .Values.webConfiguration.basicAuthUsers }} + {{ $k }}: {{ htpasswd "" $v | trimPrefix ":"}} +{{- end }} +{{- end }} + +{{/* +Define Authorization +*/}} +{{- define "prometheus-pushgateway.Authorization" -}} +{{- $users := keys .Values.webConfiguration.basicAuthUsers }} +{{- $user := first $users }} +{{- $password := index .Values.webConfiguration.basicAuthUsers $user }} +{{- $user }}:{{ $password }} +{{- end }} + +{{/* +Returns pod spec +*/}} +{{- define "prometheus-pushgateway.podSpec" -}} +serviceAccountName: {{ include "prometheus-pushgateway.serviceAccountName" . }} +automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} +{{- with .Values.priorityClassName }} +priorityClassName: {{ . | quote }} +{{- end }} +{{- with .Values.hostAliases }} +hostAliases: +{{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.imagePullSecrets }} +imagePullSecrets: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.extraInitContainers }} +initContainers: + {{- toYaml . | nindent 2 }} +{{- end }} +containers: + {{- with .Values.extraContainers }} + {{- toYaml . | nindent 2 }} + {{- end }} + - name: pushgateway + image: "{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + {{- with .Values.extraVars }} + env: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- if or .Values.extraArgs .Values.webConfiguration }} + args: + {{- with .Values.extraArgs }} + {{- toYaml . | nindent 6 }} + {{- end }} + {{- if .Values.webConfiguration }} + - --web.config.file=/etc/config/web-config.yaml + {{- end }} + {{- end }} + ports: + - name: metrics + containerPort: 9091 + protocol: TCP + {{- if .Values.liveness.enabled }} + {{- $livenessCommon := omit .Values.liveness.probe "httpGet" }} + livenessProbe: + {{- with .Values.liveness.probe }} + httpGet: + path: {{ .httpGet.path }} + port: {{ .httpGet.port }} + {{- if or .httpGet.httpHeaders $.Values.webConfiguration.basicAuthUsers }} + httpHeaders: + {{- if $.Values.webConfiguration.basicAuthUsers }} + - name: Authorization + value: Basic {{ include "prometheus-pushgateway.Authorization" $ | b64enc }} + {{- end }} + {{- with .httpGet.httpHeaders }} + {{- toYaml . | nindent 10 }} + {{- end }} + {{- end }} + {{- toYaml $livenessCommon | nindent 6 }} + {{- end }} + {{- end }} + {{- if .Values.readiness.enabled }} + {{- $readinessCommon := omit .Values.readiness.probe "httpGet" }} + readinessProbe: + {{- with .Values.readiness.probe }} + httpGet: + path: {{ .httpGet.path }} + port: {{ .httpGet.port }} + {{- if or .httpGet.httpHeaders $.Values.webConfiguration.basicAuthUsers }} + httpHeaders: + {{- if $.Values.webConfiguration.basicAuthUsers }} + - name: Authorization + value: Basic {{ include "prometheus-pushgateway.Authorization" $ | b64enc }} + {{- end }} + {{- with .httpGet.httpHeaders }} + {{- toYaml . | nindent 10 }} + {{- end }} + {{- end }} + {{- toYaml $readinessCommon | nindent 6 }} + {{- end }} + {{- end }} + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 6 }} + {{- end }} + volumeMounts: + - name: storage-volume + mountPath: "{{ .Values.persistentVolume.mountPath }}" + subPath: "{{ .Values.persistentVolume.subPath }}" + {{- if .Values.webConfiguration }} + - name: web-config + mountPath: "/etc/config" + {{- end }} + {{- with .Values.extraVolumeMounts }} + {{- toYaml . | nindent 6 }} + {{- end }} +{{- with .Values.nodeSelector }} +nodeSelector: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.tolerations }} +tolerations: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- if or .Values.podAntiAffinity .Values.affinity }} +affinity: +{{- end }} + {{- with .Values.affinity }} + {{- toYaml . | nindent 2 }} + {{- end }} + {{- if eq .Values.podAntiAffinity "hard" }} + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: {{ .Values.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ include "prometheus-pushgateway.name" . }}]} + {{- else if eq .Values.podAntiAffinity "soft" }} + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + topologyKey: {{ .Values.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ include "prometheus-pushgateway.name" . }}]} + {{- end }} +{{- with .Values.topologySpreadConstraints }} +topologySpreadConstraints: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- with .Values.securityContext }} +securityContext: + {{- toYaml . | nindent 2 }} +{{- end }} +volumes: + {{- $storageVolumeAsPVCTemplate := and .Values.runAsStatefulSet .Values.persistentVolume.enabled -}} + {{- if not $storageVolumeAsPVCTemplate }} + - name: storage-volume + {{- if .Values.persistentVolume.enabled }} + persistentVolumeClaim: + claimName: {{ if .Values.persistentVolume.existingClaim }}{{ .Values.persistentVolume.existingClaim }}{{- else }}{{ include "prometheus-pushgateway.fullname" . }}{{- end }} + {{- else }} + emptyDir: {} + {{- end }} + {{- if .Values.webConfiguration }} + - name: web-config + secret: + secretName: {{ include "prometheus-pushgateway.fullname" . }} + {{- end }} + {{- end }} + {{- if .Values.extraVolumes }} + {{- toYaml .Values.extraVolumes | nindent 2 }} + {{- else if $storageVolumeAsPVCTemplate }} + {{- if .Values.webConfiguration }} + - name: web-config + secret: + secretName: {{ include "prometheus-pushgateway.fullname" . }} + {{- else }} + [] + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml new file mode 100644 index 000000000..5d3fafde7 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml @@ -0,0 +1,29 @@ +{{- if not .Values.runAsStatefulSet }} +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + replicas: {{ .Values.replicaCount }} + {{- with .Values.strategy }} + strategy: + {{- toYaml . | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 6 }} + template: + metadata: + {{- with .Values.podAnnotations }} + annotations: + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 8 }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + spec: + {{- include "prometheus-pushgateway.podSpec" . | nindent 6 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml new file mode 100644 index 000000000..bafee9518 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml @@ -0,0 +1,8 @@ +{{- range .Values.extraManifests }} +--- + {{- if typeIs "string" . }} + {{- tpl . $ }} + {{- else }} + {{- tpl (. | toYaml | nindent 0) $ }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml new file mode 100644 index 000000000..237ac4a12 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml @@ -0,0 +1,50 @@ +{{- if .Values.ingress.enabled }} +{{- $serviceName := include "prometheus-pushgateway.fullname" . }} +{{- $servicePort := .Values.service.port }} +{{- $ingressPath := .Values.ingress.path }} +{{- $ingressClassName := .Values.ingress.className }} +{{- $ingressPathType := .Values.ingress.pathType }} +{{- $extraPaths := .Values.ingress.extraPaths }} +apiVersion: {{ include "prometheus-pushgateway.ingress.apiVersion" . }} +kind: Ingress +metadata: + {{- with .Values.ingress.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + {{- if semverCompare ">=1.19-0" .Capabilities.KubeVersion.GitVersion }} + ingressClassName: {{ $ingressClassName }} + {{- end }} + rules: + {{- range $host := .Values.ingress.hosts }} + - host: {{ $host }} + http: + paths: + {{- with $extraPaths }} + {{- toYaml . | nindent 10 }} + {{- end }} + - path: {{ $ingressPath }} + {{- if semverCompare ">=1.19-0" $.Capabilities.KubeVersion.GitVersion }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if semverCompare ">=1.19-0" $.Capabilities.KubeVersion.GitVersion }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end -}} + {{- with .Values.ingress.tls }} + tls: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml new file mode 100644 index 000000000..d3b8019e3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml @@ -0,0 +1,26 @@ +{{- if .Values.networkPolicy }} +apiVersion: {{ include "prometheus-pushgateway.networkPolicy.apiVersion" . }} +kind: NetworkPolicy +metadata: + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + {{- if .Values.networkPolicy.customSelectors }} + name: ingress-allow-customselector-{{ template "prometheus-pushgateway.name" . }} + {{- else if .Values.networkPolicy.allowAll }} + name: ingress-allow-all-{{ template "prometheus-pushgateway.name" . }} + {{- else -}} + {{- fail "One of `allowAll` or `customSelectors` must be specified." }} + {{- end }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + podSelector: + matchLabels: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 6 }} + ingress: + - ports: + - port: {{ .Values.service.targetPort }} + {{- with .Values.networkPolicy.customSelectors }} + from: + {{- toYaml . | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml new file mode 100644 index 000000000..6051133c6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.podDisruptionBudget }} +apiVersion: {{ include "prometheus-pushgateway.pdb.apiVersion" . }} +kind: PodDisruptionBudget +metadata: + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + selector: + matchLabels: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 6 }} + {{- toYaml .Values.podDisruptionBudget | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml new file mode 100644 index 000000000..d2a85f424 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml @@ -0,0 +1,29 @@ +{{- if and (not .Values.runAsStatefulSet) .Values.persistentVolume.enabled (not .Values.persistentVolume.existingClaim) }} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + {{- with .Values.persistentVolume.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + {{- with .Values.persistentVolumeLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + accessModes: + {{- toYaml .Values.persistentVolume.accessModes | nindent 4 }} + {{- if .Values.persistentVolume.storageClass }} + {{- if (eq "-" .Values.persistentVolume.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.persistentVolume.storageClass }}" + {{- end }} + {{- end }} + resources: + requests: + storage: "{{ .Values.persistentVolume.size }}" +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml new file mode 100644 index 000000000..a8142d138 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml @@ -0,0 +1,10 @@ +{{- if .Values.webConfiguration }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "prometheus-pushgateway.fullname" . }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} +data: + web-config.yaml: {{ include "prometheus-pushgateway.webConfiguration" . | b64enc}} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml new file mode 100644 index 000000000..15029f7e3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml @@ -0,0 +1,45 @@ +{{- $stsNoHeadlessSvcTypes := list "LoadBalancer" "NodePort" -}} +apiVersion: v1 +kind: Service +metadata: + {{- with .Values.serviceAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + {{- with .Values.serviceLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + {{- if .Values.service.clusterIP }} + clusterIP: {{ .Values.service.clusterIP }} + {{ else if and .Values.runAsStatefulSet (not (has .Values.service.type $stsNoHeadlessSvcTypes)) }} + clusterIP: None # Headless service + {{- end }} + {{- if .Values.service.ipDualStack.enabled }} + ipFamilies: {{ toYaml .Values.service.ipDualStack.ipFamilies | nindent 4 }} + ipFamilyPolicy: {{ .Values.service.ipDualStack.ipFamilyPolicy }} + {{- end }} + type: {{ .Values.service.type }} + {{- with .Values.service.loadBalancerIP }} + loadBalancerIP: {{ . }} + {{- end }} + {{- if .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} + {{- end }} + ports: + - port: {{ .Values.service.port }} + targetPort: {{ .Values.service.targetPort }} + {{- if and (eq .Values.service.type "NodePort") .Values.service.nodePort }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + protocol: TCP + name: {{ .Values.service.portName }} + selector: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 4 }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml new file mode 100644 index 000000000..88f147048 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml @@ -0,0 +1,17 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + {{- with .Values.serviceAccountLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ include "prometheus-pushgateway.serviceAccountName" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +automountServiceAccountToken: {{ .Values.automountServiceAccountToken }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml new file mode 100644 index 000000000..ae173199b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml @@ -0,0 +1,51 @@ +{{- if .Values.serviceMonitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + {{- if .Values.serviceMonitor.additionalLabels }} + {{- toYaml .Values.serviceMonitor.additionalLabels | nindent 4 }} + {{- end }} + name: {{ include "prometheus-pushgateway.fullname" . }} + {{- if .Values.serviceMonitor.namespace }} + namespace: {{ .Values.serviceMonitor.namespace }} + {{- else }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} + {{- end }} +spec: + endpoints: + - port: {{ .Values.service.portName }} + {{- with .Values.serviceMonitor.interval }} + interval: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.scheme }} + scheme: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.bearerTokenFile }} + bearerTokenFile: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.tlsConfig }} + tlsConfig: + {{- toYaml .| nindent 6 }} + {{- end }} + {{- with .Values.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + path: {{ .Values.serviceMonitor.telemetryPath }} + honorLabels: {{ .Values.serviceMonitor.honorLabels }} + {{- with .Values.serviceMonitor.metricRelabelings }} + metricRelabelings: + {{- tpl (toYaml . | nindent 6) $ }} + {{- end }} + {{- with .Values.serviceMonitor.relabelings }} + relabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + namespaceSelector: + matchNames: + - {{ template "prometheus-pushgateway.namespace" . }} + selector: + matchLabels: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 6 }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml new file mode 100644 index 000000000..8d486a306 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml @@ -0,0 +1,49 @@ +{{- if .Values.runAsStatefulSet }} +apiVersion: apps/v1 +kind: StatefulSet +metadata: + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 4 }} + name: {{ include "prometheus-pushgateway.fullname" . }} + namespace: {{ template "prometheus-pushgateway.namespace" . }} +spec: + replicas: {{ .Values.replicaCount }} + serviceName: {{ include "prometheus-pushgateway.fullname" . }} + selector: + matchLabels: + {{- include "prometheus-pushgateway.selectorLabels" . | nindent 6 }} + template: + metadata: + {{- with .Values.podAnnotations }} + annotations: + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 8 }} + spec: + {{- include "prometheus-pushgateway.podSpec" . | nindent 6 }} + {{- if .Values.persistentVolume.enabled }} + volumeClaimTemplates: + - metadata: + {{- with .Values.persistentVolume.annotations }} + annotations: + {{- toYaml . | nindent 10 }} + {{- end }} + labels: + {{- include "prometheus-pushgateway.defaultLabels" . | nindent 10 }} + name: storage-volume + spec: + accessModes: + {{ toYaml .Values.persistentVolume.accessModes }} + {{- if .Values.persistentVolume.storageClass }} + {{- if (eq "-" .Values.persistentVolume.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.persistentVolume.storageClass }}" + {{- end }} + {{- end }} + resources: + requests: + storage: "{{ .Values.persistentVolume.size }}" + {{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/values.yaml new file mode 100644 index 000000000..85f267fdb --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/charts/prometheus-pushgateway/values.yaml @@ -0,0 +1,371 @@ +# Default values for prometheus-pushgateway. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# Provide a name in place of prometheus-pushgateway for `app:` labels +nameOverride: "" + +# Provide a name to substitute for the full names of resources +fullnameOverride: "" + +# Provide a namespace to substitude for the namespace on resources +namespaceOverride: "" + +image: + repository: quay.io/prometheus/pushgateway + # if not set appVersion field from Chart.yaml is used + tag: "" + pullPolicy: IfNotPresent + +# Optional pod imagePullSecrets +imagePullSecrets: [] + +service: + type: ClusterIP + port: 9091 + targetPort: 9091 + # nodePort: 32100 + portName: http + + # Optional - Can be used for headless if value is "None" + clusterIP: "" + + ipDualStack: + enabled: false + ipFamilies: ["IPv6", "IPv4"] + ipFamilyPolicy: "PreferDualStack" + + loadBalancerIP: "" + loadBalancerSourceRanges: [] + +# Whether to automatically mount a service account token into the pod +automountServiceAccountToken: true + +# Optional pod annotations +podAnnotations: {} + +# Optional pod labels +podLabels: {} + +# Optional service annotations +serviceAnnotations: {} + +# Optional service labels +serviceLabels: {} + +# Optional serviceAccount labels +serviceAccountLabels: {} + +# Optional persistentVolume labels +persistentVolumeLabels: {} + +# Optional additional environment variables +extraVars: [] + +## Additional pushgateway container arguments +## +## example: +## extraArgs: +## - --persistence.file=/data/pushgateway.data +## - --persistence.interval=5m +extraArgs: [] + +## Additional InitContainers to initialize the pod +## +extraInitContainers: [] + +# Optional additional containers (sidecar) +extraContainers: [] + # - name: oAuth2-proxy + # args: + # - -https-address=:9092 + # - -upstream=http://localhost:9091 + # - -skip-auth-regex=^/metrics + # - -openshift-delegate-urls={"/":{"group":"monitoring.coreos.com","resource":"prometheuses","verb":"get"}} + # image: openshift/oauth-proxy:v1.1.0 + # ports: + # - containerPort: 9092 + # name: proxy + # resources: + # limits: + # memory: 16Mi + # requests: + # memory: 4Mi + # cpu: 20m + # volumeMounts: + # - mountPath: /etc/prometheus/secrets/pushgateway-tls + # name: secret-pushgateway-tls + +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: 200m + # memory: 50Mi + # requests: + # cpu: 100m + # memory: 30Mi + +# -- Sets web configuration +# To enable basic authentication, provide basicAuthUsers as a map +webConfiguration: {} + # basicAuthUsers: + # username: password + +liveness: + enabled: true + probe: + httpGet: + path: /-/healthy + port: 9091 + initialDelaySeconds: 10 + timeoutSeconds: 10 + +readiness: + enabled: true + probe: + httpGet: + path: /-/ready + port: 9091 + initialDelaySeconds: 10 + timeoutSeconds: 10 + +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: + +## Configure ingress resource that allow you to access the +## pushgateway installation. Set up the URL +## ref: http://kubernetes.io/docs/user-guide/ingress/ +## +ingress: + ## Enable Ingress. + ## + enabled: false + # AWS ALB requires path of /* + className: "" + path: / + pathType: ImplementationSpecific + + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + + ## Annotations. + ## + # annotations: + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: 'true' + + ## Hostnames. + ## Must be provided if Ingress is enabled. + ## + # hosts: + # - pushgateway.domain.com + + ## TLS configuration. + ## Secrets must be manually created in the namespace. + ## + # tls: + # - secretName: pushgateway-tls + # hosts: + # - pushgateway.domain.com + +tolerations: [] + # - effect: NoSchedule + # operator: Exists + +## Node labels for pushgateway pod assignment +## Ref: https://kubernetes.io/docs/user-guide/node-selection/ +## +nodeSelector: {} + +replicaCount: 1 + +hostAliases: [] + # - ip: "127.0.0.1" + # hostnames: + # - "foo.local" + # - "bar.local" + # - ip: "10.1.2.3" + # hostnames: + # - "foo.remote" + # - "bar.remote" + +## When running more than one replica alongside with persistence, different volumes are needed +## per replica, since sharing a `persistence.file` across replicas does not keep metrics synced. +## For this purpose, you can enable the `runAsStatefulSet` to deploy the pushgateway as a +## StatefulSet instead of as a Deployment. +runAsStatefulSet: false + +## Security context to be added to push-gateway pods +## +securityContext: + fsGroup: 65534 + runAsUser: 65534 + runAsNonRoot: true + +## Security context to be added to push-gateway containers +## Having a separate variable as securityContext differs for pods and containers. +containerSecurityContext: {} +# allowPrivilegeEscalation: false +# readOnlyRootFilesystem: true +# runAsUser: 65534 +# runAsNonRoot: true + +## Affinity for pod assignment +## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity +affinity: {} + +## Pod anti-affinity can prevent the scheduler from placing pushgateway replicas on the same node. +## The value "soft" means that the scheduler should *prefer* to not schedule two replica pods onto the same node but no guarantee is provided. +## The value "hard" means that the scheduler is *required* to not schedule two replica pods onto the same node. +## The default value "" will disable pod anti-affinity so that no anti-affinity rules will be configured (unless set in `affinity`). +## +podAntiAffinity: "" + +## If anti-affinity is enabled sets the topologyKey to use for anti-affinity. +## This can be changed to, for example, failure-domain.beta.kubernetes.io/zone +## +podAntiAffinityTopologyKey: kubernetes.io/hostname + +## Topology spread constraints for pods +## Ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ +topologySpreadConstraints: [] + +# Enable this if you're using https://github.com/coreos/prometheus-operator +serviceMonitor: + enabled: false + namespace: monitoring + + # telemetryPath: HTTP resource path from which to fetch metrics. + # Telemetry path, default /metrics, has to be prefixed accordingly if pushgateway sets a route prefix at start-up. + # + telemetryPath: "/metrics" + + # Fallback to the prometheus default unless specified + # interval: 10s + + ## scheme: HTTP scheme to use for scraping. Can be used with `tlsConfig` for example if using istio mTLS. + # scheme: "" + + ## tlsConfig: TLS configuration to use when scraping the endpoint. For example if using istio mTLS. + ## Of type: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#tlsconfig + # tlsConfig: {} + + # bearerTokenFile: + # Fallback to the prometheus default unless specified + # scrapeTimeout: 30s + + ## Used to pass Labels that are used by the Prometheus installed in your cluster to select Service Monitors to work with + ## ref: https://github.com/coreos/prometheus-operator/blob/master/Documentation/api.md#prometheusspec + additionalLabels: {} + + # Retain the job and instance labels of the metrics pushed to the Pushgateway + # [Scraping Pushgateway](https://github.com/prometheus/pushgateway#configure-the-pushgateway-as-a-target-to-scrape) + honorLabels: true + + ## Metric relabel configs to apply to samples before ingestion. + ## [Metric Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#metric_relabel_configs) + metricRelabelings: [] + # - action: keep + # regex: 'kube_(daemonset|deployment|pod|namespace|node|statefulset).+' + # sourceLabels: [__name__] + + ## Relabel configs to apply to samples before ingestion. + ## [Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config) + relabelings: [] + # - sourceLabels: [__meta_kubernetes_pod_node_name] + # separator: ; + # regex: ^(.*)$ + # targetLabel: nodename + # replacement: $1 + # action: replace + +# The values to set in the PodDisruptionBudget spec (minAvailable/maxUnavailable) +# If not set then a PodDisruptionBudget will not be created +podDisruptionBudget: {} + +priorityClassName: + +# Deployment Strategy type +strategy: + type: Recreate + +persistentVolume: + ## If true, pushgateway will create/use a Persistent Volume Claim + ## If false, use emptyDir + ## + enabled: false + + ## pushgateway data Persistent Volume access modes + ## Must match those of existing PV or dynamic provisioner + ## Ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ + ## + accessModes: + - ReadWriteOnce + + ## pushgateway data Persistent Volume Claim annotations + ## + annotations: {} + + ## pushgateway data Persistent Volume existing claim name + ## Requires pushgateway.persistentVolume.enabled: true + ## If defined, PVC must be created manually before volume will be bound + existingClaim: "" + + ## pushgateway data Persistent Volume mount root path + ## + mountPath: /data + + ## pushgateway data Persistent Volume size + ## + size: 2Gi + + ## pushgateway data Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + # storageClass: "-" + + ## Subdirectory of pushgateway data Persistent Volume to mount + ## Useful if the volume's root directory is not empty + ## + subPath: "" + +extraVolumes: [] + # - name: extra + # emptyDir: {} +extraVolumeMounts: [] + # - name: extra + # mountPath: /usr/share/extras + # readOnly: true + +# Configuration for clusters with restrictive network policies in place: +# - allowAll allows access to the PushGateway from any namespace +# - customSelector is a list of pod/namespaceSelectors to allow access from +# These options are mutually exclusive and the latter will take precedence. +networkPolicy: {} + # allowAll: true + # customSelectors: + # - namespaceSelector: + # matchLabels: + # type: admin + # - podSelector: + # matchLabels: + # app: myapp + +# Array of extra K8s objects to deploy (evaluated as a template) +# The value can hold an array of strings as well as objects +extraManifests: [] diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/NOTES.txt new file mode 100644 index 000000000..fc03c2a5b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/NOTES.txt @@ -0,0 +1,113 @@ +The Prometheus server can be accessed via port {{ .Values.server.service.servicePort }} on the following DNS name from within your cluster: +{{ template "prometheus.server.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if .Values.server.ingress.enabled -}} +From outside the cluster, the server URL(s) are: +{{- range .Values.server.ingress.hosts }} +http://{{ . }} +{{- end }} +{{- else }} +Get the Prometheus server URL by running these commands in the same shell: +{{- if contains "NodePort" .Values.server.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus.server.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.server.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "prometheus.server.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "prometheus.server.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.server.service.servicePort }} +{{- else if contains "ClusterIP" .Values.server.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app.kubernetes.io/name={{ include "prometheus.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9090 +{{- end }} + + +{{- if .Values.server.persistentVolume.enabled }} +{{- else }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the Server pod is terminated. ##### +################################################################################# +{{- end }} +{{- end }} + +{{ if .Values.alertmanager.enabled }} +The Prometheus alertmanager can be accessed via port {{ .Values.alertmanager.service.port }} on the following DNS name from within your cluster: +{{ template "prometheus.alertmanager.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if .Values.alertmanager.ingress.enabled -}} +From outside the cluster, the alertmanager URL(s) are: +{{- range .Values.alertmanager.ingress.hosts }} +http://{{ . }} +{{- end }} +{{- else }} +Get the Alertmanager URL by running these commands in the same shell: +{{- if contains "NodePort" .Values.alertmanager.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus.alertmanager.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.alertmanager.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "prometheus.alertmanager.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "prometheus.alertmanager.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.alertmanager.service.servicePort }} +{{- else if contains "ClusterIP" .Values.alertmanager.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app.kubernetes.io/name={{ include "alertmanager.name" .Subcharts.alertmanager }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9093 +{{- end }} +{{- end }} + +{{- if .Values.alertmanager.persistence.enabled }} +{{- else }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the AlertManager pod is terminated. ##### +################################################################################# +{{- end }} +{{- end }} + +{{- if (index .Values "prometheus-node-exporter" "enabled") }} +################################################################################# +###### WARNING: Pod Security Policy has been disabled by default since ##### +###### it deprecated after k8s 1.25+. use ##### +###### (index .Values "prometheus-node-exporter" "rbac" ##### +###### . "pspEnabled") with (index .Values ##### +###### "prometheus-node-exporter" "rbac" "pspAnnotations") ##### +###### in case you still need it. ##### +################################################################################# +{{- end }} + +{{ if (index .Values "prometheus-pushgateway" "enabled") }} +The Prometheus PushGateway can be accessed via port {{ index .Values "prometheus-pushgateway" "service" "port" }} on the following DNS name from within your cluster: +{{ include "prometheus-pushgateway.fullname" (index .Subcharts "prometheus-pushgateway") }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if (index .Values "prometheus-pushgateway" "ingress" "enabled") -}} +From outside the cluster, the pushgateway URL(s) are: +{{- range (index .Values "prometheus-pushgateway" "ingress" "hosts") }} +http://{{ . }} +{{- end }} +{{- else }} +Get the PushGateway URL by running these commands in the same shell: +{{- $pushgateway_svc_type := index .Values "prometheus-pushgateway" "service" "type" -}} +{{- if contains "NodePort" $pushgateway_svc_type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "prometheus-pushgateway.fullname" (index .Subcharts "prometheus-pushgateway") }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" $pushgateway_svc_type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ include "prometheus-pushgateway.fullname" (index .Subcharts "prometheus-pushgateway") }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "prometheus-pushgateway.fullname" (index .Subcharts "prometheus-pushgateway") }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ index .Values "prometheus-pushgateway" "service" "port" }} +{{- else if contains "ClusterIP" $pushgateway_svc_type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app={{ include "prometheus.name" (index .Subcharts "prometheus-pushgateway") }},component=pushgateway" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9091 +{{- end }} +{{- end }} +{{- end }} + +For more information on running Prometheus, visit: +https://prometheus.io/ diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/_helpers.tpl new file mode 100644 index 000000000..3d8078f02 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/_helpers.tpl @@ -0,0 +1,238 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "prometheus.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "prometheus.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create labels for prometheus +*/}} +{{- define "prometheus.common.matchLabels" -}} +app.kubernetes.io/name: {{ include "prometheus.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end -}} + +{{/* +Create unified labels for prometheus components +*/}} +{{- define "prometheus.common.metaLabels" -}} +app.kubernetes.io/version: {{ .Chart.AppVersion }} +helm.sh/chart: {{ include "prometheus.chart" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/part-of: {{ include "prometheus.name" . }} +{{- with .Values.commonMetaLabels}} +{{ toYaml . }} +{{- end }} +{{- end -}} + +{{- define "prometheus.server.labels" -}} +{{ include "prometheus.server.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +{{- end -}} + +{{- define "prometheus.server.matchLabels" -}} +app.kubernetes.io/component: {{ .Values.server.name }} +{{ include "prometheus.common.matchLabels" . }} +{{- 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 "prometheus.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 a fully qualified ClusterRole name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "prometheus.clusterRoleName" -}} +{{- if .Values.server.clusterRoleNameOverride -}} +{{ .Values.server.clusterRoleNameOverride | trunc 63 | trimSuffix "-" }} +{{- else -}} +{{ include "prometheus.server.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Create a fully qualified alertmanager name for communicating and check to ensure that `alertmanager` exists before trying to use it with the user via NOTES.txt +*/}} +{{- define "prometheus.alertmanager.fullname" -}} +{{- if .Subcharts.alertmanager -}} +{{- template "alertmanager.fullname" .Subcharts.alertmanager -}} +{{- else -}} +{{- "alertmanager not found" -}} +{{- end -}} +{{- end -}} + +{{/* +Create a fully qualified Prometheus server name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "prometheus.server.fullname" -}} +{{- if .Values.server.fullnameOverride -}} +{{- .Values.server.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name .Values.server.name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.server.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Get KubeVersion removing pre-release information. +*/}} +{{- define "prometheus.kubeVersion" -}} + {{- default .Capabilities.KubeVersion.Version (regexFind "v[0-9]+\\.[0-9]+\\.[0-9]+" .Capabilities.KubeVersion.Version) -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for deployment. +*/}} +{{- define "prometheus.deployment.apiVersion" -}} +{{- print "apps/v1" -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for networkpolicy. +*/}} +{{- define "prometheus.networkPolicy.apiVersion" -}} +{{- print "networking.k8s.io/v1" -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for poddisruptionbudget. +*/}} +{{- define "prometheus.podDisruptionBudget.apiVersion" -}} +{{- if .Capabilities.APIVersions.Has "policy/v1" }} +{{- print "policy/v1" -}} +{{- else -}} +{{- print "policy/v1beta1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for rbac. +*/}} +{{- define "rbac.apiVersion" -}} +{{- if .Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1" }} +{{- print "rbac.authorization.k8s.io/v1" -}} +{{- else -}} +{{- print "rbac.authorization.k8s.io/v1beta1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "ingress.apiVersion" -}} + {{- if and (.Capabilities.APIVersions.Has "networking.k8s.io/v1") (semverCompare ">= 1.19.x" (include "prometheus.kubeVersion" .)) -}} + {{- print "networking.k8s.io/v1" -}} + {{- else if .Capabilities.APIVersions.Has "networking.k8s.io/v1beta1" -}} + {{- print "networking.k8s.io/v1beta1" -}} + {{- else -}} + {{- print "extensions/v1beta1" -}} + {{- end -}} +{{- end -}} + +{{/* +Return if ingress is stable. +*/}} +{{- define "ingress.isStable" -}} + {{- eq (include "ingress.apiVersion" .) "networking.k8s.io/v1" -}} +{{- end -}} + +{{/* +Return if ingress supports ingressClassName. +*/}} +{{- define "ingress.supportsIngressClassName" -}} + {{- or (eq (include "ingress.isStable" .) "true") (and (eq (include "ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18.x" (include "prometheus.kubeVersion" .))) -}} +{{- end -}} + +{{/* +Return if ingress supports pathType. +*/}} +{{- define "ingress.supportsPathType" -}} + {{- or (eq (include "ingress.isStable" .) "true") (and (eq (include "ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18.x" (include "prometheus.kubeVersion" .))) -}} +{{- end -}} + +{{/* +Create the name of the service account to use for the server component +*/}} +{{- define "prometheus.serviceAccountName.server" -}} +{{- if .Values.serviceAccounts.server.create -}} + {{ default (include "prometheus.server.fullname" .) .Values.serviceAccounts.server.name }} +{{- else -}} + {{ default "default" .Values.serviceAccounts.server.name }} +{{- end -}} +{{- end -}} + +{{/* +Define the prometheus.namespace template if set with forceNamespace or .Release.Namespace is set +*/}} +{{- define "prometheus.namespace" -}} + {{- default .Release.Namespace .Values.forceNamespace -}} +{{- end }} + +{{/* +Define template prometheus.namespaces producing a list of namespaces to monitor +*/}} +{{- define "prometheus.namespaces" -}} +{{- $namespaces := list }} +{{- if and .Values.rbac.create .Values.server.useExistingClusterRoleName }} + {{- if .Values.server.namespaces -}} + {{- range $ns := join "," .Values.server.namespaces | split "," }} + {{- $namespaces = append $namespaces (tpl $ns $) }} + {{- end -}} + {{- end -}} + {{- if .Values.server.releaseNamespace -}} + {{- $namespaces = append $namespaces (include "prometheus.namespace" .) }} + {{- end -}} +{{- end -}} +{{ mustToJson $namespaces }} +{{- end -}} + +{{/* +Define prometheus.server.remoteWrite producing a list of remoteWrite configurations with URL templating +*/}} +{{- define "prometheus.server.remoteWrite" -}} +{{- $remoteWrites := list }} +{{- range $remoteWrite := .Values.server.remoteWrite }} + {{- $remoteWrites = tpl $remoteWrite.url $ | set $remoteWrite "url" | append $remoteWrites }} +{{- end -}} +{{ toYaml $remoteWrites }} +{{- end -}} + +{{/* +Define prometheus.server.remoteRead producing a list of remoteRead configurations with URL templating +*/}} +{{- define "prometheus.server.remoteRead" -}} +{{- $remoteReads := list }} +{{- range $remoteRead := .Values.server.remoteRead }} + {{- $remoteReads = tpl $remoteRead.url $ | set $remoteRead "url" | append $remoteReads }} +{{- end -}} +{{ toYaml $remoteReads }} +{{- end -}} + diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrole.yaml new file mode 100644 index 000000000..25e3cec45 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrole.yaml @@ -0,0 +1,56 @@ +{{- if and .Values.rbac.create (empty .Values.server.useExistingClusterRoleName) -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRole +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ include "prometheus.clusterRoleName" . }} +rules: +{{- if and .Values.podSecurityPolicy.enabled (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ template "prometheus.server.fullname" . }} +{{- end }} + - apiGroups: + - "" + resources: + - nodes + - nodes/proxy + - nodes/metrics + - services + - endpoints + - pods + - ingresses + - configmaps + verbs: + - get + - list + - watch + - apiGroups: + - "extensions" + - "networking.k8s.io" + resources: + - ingresses/status + - ingresses + verbs: + - get + - list + - watch + - apiGroups: + - "discovery.k8s.io" + resources: + - endpointslices + verbs: + - get + - list + - watch + - nonResourceURLs: + - "/metrics" + verbs: + - get +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..28f4bda77 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.rbac.create (empty .Values.server.namespaces) (empty .Values.server.useExistingClusterRoleName) -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRoleBinding +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ include "prometheus.clusterRoleName" . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.server" . }} + namespace: {{ include "prometheus.namespace" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "prometheus.clusterRoleName" . }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/cm.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/cm.yaml new file mode 100644 index 000000000..8713bd1ea --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/cm.yaml @@ -0,0 +1,103 @@ +{{- if (empty .Values.server.configMapOverrideName) -}} +apiVersion: v1 +kind: ConfigMap +metadata: +{{- with .Values.server.configMapAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + {{- with .Values.server.extraConfigmapLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +data: + allow-snippet-annotations: "false" +{{- $root := . -}} +{{- range $key, $value := .Values.ruleFiles }} + {{ $key }}: {{- toYaml $value | indent 2 }} +{{- end }} +{{- range $key, $value := .Values.serverFiles }} + {{ $key }}: | +{{- if eq $key "prometheus.yml" }} + global: +{{ $root.Values.server.global | toYaml | trimSuffix "\n" | indent 6 }} +{{- if $root.Values.server.remoteWrite }} + remote_write: +{{- include "prometheus.server.remoteWrite" $root | nindent 4 }} +{{- end }} +{{- if $root.Values.server.remoteRead }} + remote_read: +{{- include "prometheus.server.remoteRead" $root | nindent 4 }} +{{- end }} +{{- if or $root.Values.server.tsdb $root.Values.server.exemplars }} + storage: +{{- if $root.Values.server.tsdb }} + tsdb: +{{ $root.Values.server.tsdb | toYaml | indent 8 }} +{{- end }} +{{- if $root.Values.server.exemplars }} + exemplars: +{{ $root.Values.server.exemplars | toYaml | indent 8 }} +{{- end }} +{{- end }} +{{- if $root.Values.scrapeConfigFiles }} + scrape_config_files: +{{ toYaml $root.Values.scrapeConfigFiles | indent 4 }} +{{- end }} +{{- end }} +{{- if eq $key "alerts" }} +{{- if and (not (empty $value)) (empty $value.groups) }} + groups: +{{- range $ruleKey, $ruleValue := $value }} + - name: {{ $ruleKey -}}.rules + rules: +{{ $ruleValue | toYaml | trimSuffix "\n" | indent 6 }} +{{- end }} +{{- else }} +{{ toYaml $value | indent 4 }} +{{- end }} +{{- else }} +{{ toYaml $value | default "{}" | indent 4 }} +{{- end }} +{{- if eq $key "prometheus.yml" -}} +{{- if $root.Values.extraScrapeConfigs }} +{{ tpl $root.Values.extraScrapeConfigs $root | indent 4 }} +{{- end -}} +{{- if or ($root.Values.alertmanager.enabled) ($root.Values.server.alertmanagers) }} + alerting: +{{- if $root.Values.alertRelabelConfigs }} +{{ $root.Values.alertRelabelConfigs | toYaml | trimSuffix "\n" | indent 6 }} +{{- end }} + alertmanagers: +{{- if $root.Values.server.alertmanagers }} +{{ toYaml $root.Values.server.alertmanagers | indent 8 }} +{{- else }} + - kubernetes_sd_configs: + - role: pod + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + {{- if $root.Values.alertmanager.prefixURL }} + path_prefix: {{ $root.Values.alertmanager.prefixURL }} + {{- end }} + relabel_configs: + - source_labels: [__meta_kubernetes_namespace] + regex: {{ $root.Release.Namespace }} + action: keep + - source_labels: [__meta_kubernetes_pod_label_app_kubernetes_io_instance] + regex: {{ $root.Release.Name }} + action: keep + - source_labels: [__meta_kubernetes_pod_label_app_kubernetes_io_name] + regex: {{ default "alertmanager" $root.Values.alertmanager.nameOverride | trunc 63 | trimSuffix "-" }} + action: keep + - source_labels: [__meta_kubernetes_pod_container_port_number] + regex: "9093" + action: keep +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/deploy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/deploy.yaml new file mode 100644 index 000000000..9c786ee42 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/deploy.yaml @@ -0,0 +1,410 @@ +{{- if not .Values.server.statefulSet.enabled -}} +apiVersion: {{ template "prometheus.deployment.apiVersion" . }} +kind: Deployment +metadata: +{{- if .Values.server.deploymentAnnotations }} + annotations: + {{ toYaml .Values.server.deploymentAnnotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +spec: + selector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + replicas: {{ .Values.server.replicaCount }} + revisionHistoryLimit: {{ .Values.server.revisionHistoryLimit }} + {{- if .Values.server.strategy }} + strategy: +{{ toYaml .Values.server.strategy | trim | indent 4 }} + {{ if eq .Values.server.strategy.type "Recreate" }}rollingUpdate: null{{ end }} +{{- end }} + template: + metadata: + {{- if .Values.server.podAnnotations }} + annotations: + {{ toYaml .Values.server.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 8 }} + {{- if .Values.server.podLabels}} + {{ toYaml .Values.server.podLabels | nindent 8 }} + {{- end}} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + spec: +{{- if .Values.server.priorityClassName }} + priorityClassName: "{{ .Values.server.priorityClassName }}" +{{- end }} +{{- if .Values.server.schedulerName }} + schedulerName: "{{ .Values.server.schedulerName }}" +{{- end }} +{{- if semverCompare ">=1.13-0" .Capabilities.KubeVersion.GitVersion }} + {{- if or (.Values.server.enableServiceLinks) (eq (.Values.server.enableServiceLinks | toString) "") }} + enableServiceLinks: true + {{- else }} + enableServiceLinks: false + {{- end }} +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.server" . }} +{{- if kindIs "bool" .Values.server.automountServiceAccountToken }} + automountServiceAccountToken: {{ .Values.server.automountServiceAccountToken }} +{{- end }} + {{- if .Values.server.extraInitContainers }} + initContainers: +{{ toYaml .Values.server.extraInitContainers | indent 8 }} + {{- end }} + containers: + {{- if .Values.configmapReload.prometheus.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }}-{{ .Values.configmapReload.prometheus.name }} + {{- if .Values.configmapReload.prometheus.image.digest }} + image: "{{ .Values.configmapReload.prometheus.image.repository }}@{{ .Values.configmapReload.prometheus.image.digest }}" + {{- else }} + image: "{{ .Values.configmapReload.prometheus.image.repository }}:{{ .Values.configmapReload.prometheus.image.tag }}" + {{- end }} + imagePullPolicy: "{{ .Values.configmapReload.prometheus.image.pullPolicy }}" + {{- with .Values.configmapReload.prometheus.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + args: + - --watched-dir=/etc/config + {{- $default_url := "http://127.0.0.1:9090/-/reload" }} + {{- with .Values.server.prefixURL }} + {{- $default_url = printf "http://127.0.0.1:9090%s/-/reload" . }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + - --listen-address=0.0.0.0:{{ .Values.configmapReload.prometheus.containerPort }} + {{- end }} + - --reload-url={{ default $default_url .Values.configmapReload.reloadUrl }} + {{- range $key, $value := .Values.configmapReload.prometheus.extraArgs }} + {{- if $value }} + - --{{ $key }}={{ $value }} + {{- else }} + - --{{ $key }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraVolumeDirs }} + - --watched-dir={{ . }} + {{- end }} + {{- with .Values.configmapReload.env }} + env: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + ports: + - containerPort: {{ .Values.configmapReload.prometheus.containerPort }} + {{- if .Values.configmapReload.prometheus.containerPortName }} + name: {{ .Values.configmapReload.prometheus.containerPortName }} + {{- end }} + {{- end }} + {{- with .Values.configmapReload.prometheus.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.configmapReload.prometheus.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.configmapReload.prometheus.startupProbe.enabled }} + {{- $startupProbe := omit .Values.configmapReload.prometheus.startupProbe "enabled" }} + startupProbe: + {{- toYaml $startupProbe | nindent 12 }} + {{- end }} + {{- with .Values.configmapReload.prometheus.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- with .Values.configmapReload.prometheus.extraVolumeMounts }} + {{ toYaml . | nindent 12 }} + {{- end }} + {{- end }} + + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }} + {{- if .Values.server.image.digest }} + image: "{{ .Values.server.image.repository }}@{{ .Values.server.image.digest }}" + {{- else }} + image: "{{ .Values.server.image.repository }}:{{ .Values.server.image.tag | default .Chart.AppVersion}}" + {{- end }} + imagePullPolicy: "{{ .Values.server.image.pullPolicy }}" + {{- with .Values.server.command }} + command: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.server.env }} + env: +{{ toYaml .Values.server.env | indent 12}} + {{- end }} + args: + {{- if .Values.server.defaultFlagsOverride }} + {{ toYaml .Values.server.defaultFlagsOverride | nindent 12}} + {{- else }} + {{- if .Values.server.retention }} + - --storage.tsdb.retention.time={{ .Values.server.retention }} + {{- end }} + {{- if .Values.server.retentionSize }} + - --storage.tsdb.retention.size={{ .Values.server.retentionSize }} + {{- end }} + - --config.file={{ .Values.server.configPath }} + {{- if .Values.server.storagePath }} + - --storage.tsdb.path={{ .Values.server.storagePath }} + {{- else }} + - --storage.tsdb.path={{ .Values.server.persistentVolume.mountPath }} + {{- end }} + - --web.console.libraries=/etc/prometheus/console_libraries + - --web.console.templates=/etc/prometheus/consoles + {{- range .Values.server.extraFlags }} + - --{{ . }} + {{- end }} + {{- range $key, $value := .Values.server.extraArgs }} + {{- if $value }} + - --{{ $key }}={{ $value }} + {{- else }} + - --{{ $key }} + {{- end }} + {{- end }} + {{- if .Values.server.prefixURL }} + - --web.route-prefix={{ .Values.server.prefixURL }} + {{- end }} + {{- if .Values.server.baseURL }} + - --web.external-url={{ .Values.server.baseURL }} + {{- end }} + {{- end }} + ports: + - containerPort: 9090 + {{- if .Values.server.portName }} + name: {{ .Values.server.portName }} + {{- end }} + {{- if .Values.server.hostPort }} + hostPort: {{ .Values.server.hostPort }} + {{- end }} + readinessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/ready + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- with .Values.server.probeHeaders }} + httpHeaders: +{{- toYaml . | nindent 14 }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.readinessProbeInitialDelay }} + periodSeconds: {{ .Values.server.readinessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.readinessProbeTimeout }} + failureThreshold: {{ .Values.server.readinessProbeFailureThreshold }} + successThreshold: {{ .Values.server.readinessProbeSuccessThreshold }} + livenessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- with .Values.server.probeHeaders }} + httpHeaders: +{{- toYaml . | nindent 14 }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.livenessProbeInitialDelay }} + periodSeconds: {{ .Values.server.livenessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.livenessProbeTimeout }} + failureThreshold: {{ .Values.server.livenessProbeFailureThreshold }} + successThreshold: {{ .Values.server.livenessProbeSuccessThreshold }} + {{- if .Values.server.startupProbe.enabled }} + startupProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- if .Values.server.probeHeaders }} + httpHeaders: + {{- range .Values.server.probeHeaders}} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + failureThreshold: {{ .Values.server.startupProbe.failureThreshold }} + periodSeconds: {{ .Values.server.startupProbe.periodSeconds }} + timeoutSeconds: {{ .Values.server.startupProbe.timeoutSeconds }} + {{- end }} + {{- with .Values.server.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: storage-volume + mountPath: {{ .Values.server.persistentVolume.mountPath }} + subPath: "{{ .Values.server.persistentVolume.subPath }}" + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.server.extraVolumeMounts }} + {{ toYaml .Values.server.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- with .Values.server.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.server.sidecarContainers }} + {{- range $name, $spec := .Values.server.sidecarContainers }} + - name: {{ $name }} + {{- if kindIs "string" $spec }} + {{- tpl $spec $ | nindent 10 }} + {{- else }} + {{- toYaml $spec | nindent 10 }} + {{- end }} + {{- end }} + {{- end }} + {{- if .Values.server.hostNetwork }} + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet + {{- else }} + dnsPolicy: {{ .Values.server.dnsPolicy }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.server.nodeSelector }} + nodeSelector: +{{ toYaml .Values.server.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.server.hostAliases }} + hostAliases: +{{ toYaml .Values.server.hostAliases | indent 8 }} + {{- end }} + {{- if .Values.server.dnsConfig }} + dnsConfig: +{{ toYaml .Values.server.dnsConfig | indent 8 }} + {{- end }} + {{- with .Values.server.securityContext }} + securityContext: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.server.tolerations }} + tolerations: +{{ toYaml .Values.server.tolerations | indent 8 }} + {{- end }} + {{- if or .Values.server.affinity .Values.server.podAntiAffinity }} + affinity: + {{- end }} + {{- with .Values.server.affinity }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if eq .Values.server.podAntiAffinity "hard" }} + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: {{ .Values.server.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ template "prometheus.name" . }}]} + {{- else if eq .Values.server.podAntiAffinity "soft" }} + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + topologyKey: {{ .Values.server.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ template "prometheus.name" . }}]} + {{- end }} + {{- with .Values.server.topologySpreadConstraints }} + topologySpreadConstraints: + {{- toYaml . | nindent 8 }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.server.terminationGracePeriodSeconds }} + volumes: + - name: config-volume + {{- if empty .Values.server.configFromSecret }} + configMap: + name: {{ if .Values.server.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.server.configMapOverrideName }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.server.configFromSecret }} + {{- end }} + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + hostPath: + path: {{ .hostPath }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} +{{- if .Values.server.extraVolumes }} +{{ toYaml .Values.server.extraVolumes | indent 8}} +{{- end }} + - name: storage-volume + {{- if .Values.server.persistentVolume.enabled }} + persistentVolumeClaim: + claimName: {{ if .Values.server.persistentVolume.existingClaim }}{{ .Values.server.persistentVolume.existingClaim }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + emptyDir: + {{- if .Values.server.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.server.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/extra-manifests.yaml new file mode 100644 index 000000000..2b21b7106 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraManifests }} +--- +{{ tpl . $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/headless-svc.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/headless-svc.yaml new file mode 100644 index 000000000..df9db9914 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/headless-svc.yaml @@ -0,0 +1,35 @@ +{{- if .Values.server.statefulSet.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.server.statefulSet.headless.annotations }} + annotations: +{{ toYaml .Values.server.statefulSet.headless.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- if .Values.server.statefulSet.headless.labels }} +{{ toYaml .Values.server.statefulSet.headless.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }}-headless + namespace: {{ include "prometheus.namespace" . }} +spec: + clusterIP: None + ports: + - name: http + port: {{ .Values.server.statefulSet.headless.servicePort }} + protocol: TCP + targetPort: 9090 + {{- if .Values.server.statefulSet.headless.gRPC.enabled }} + - name: grpc + port: {{ .Values.server.statefulSet.headless.gRPC.servicePort }} + protocol: TCP + targetPort: 10901 + {{- if .Values.server.statefulSet.headless.gRPC.nodePort }} + nodePort: {{ .Values.server.statefulSet.headless.gRPC.nodePort }} + {{- end }} + {{- end }} + + selector: + {{- include "prometheus.server.matchLabels" . | nindent 4 }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/ingress.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/ingress.yaml new file mode 100644 index 000000000..84341a9c2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/ingress.yaml @@ -0,0 +1,57 @@ +{{- if .Values.server.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "ingress.supportsPathType" .) "true" -}} +{{- $releaseName := .Release.Name -}} +{{- $serviceName := include "prometheus.server.fullname" . }} +{{- $servicePort := .Values.server.ingress.servicePort | default .Values.server.service.servicePort -}} +{{- $ingressPath := .Values.server.ingress.path -}} +{{- $ingressPathType := .Values.server.ingress.pathType -}} +{{- $extraPaths := .Values.server.ingress.extraPaths -}} +apiVersion: {{ template "ingress.apiVersion" . }} +kind: Ingress +metadata: +{{- if .Values.server.ingress.annotations }} + annotations: +{{ toYaml .Values.server.ingress.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- range $key, $value := .Values.server.ingress.extraLabels }} + {{ $key }}: {{ $value }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.server.ingress.ingressClassName }} + ingressClassName: {{ .Values.server.ingress.ingressClassName }} + {{- end }} + rules: + {{- range .Values.server.ingress.hosts }} + {{- $url := splitList "/" . }} + - host: {{ first $url }} + http: + paths: +{{ if $extraPaths }} +{{ toYaml $extraPaths | indent 10 }} +{{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end -}} +{{- if .Values.server.ingress.tls }} + tls: +{{ toYaml .Values.server.ingress.tls | indent 4 }} + {{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/network-policy.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/network-policy.yaml new file mode 100644 index 000000000..3254ffc04 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/network-policy.yaml @@ -0,0 +1,16 @@ +{{- if .Values.networkPolicy.enabled }} +apiVersion: {{ template "prometheus.networkPolicy.apiVersion" . }} +kind: NetworkPolicy +metadata: + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + ingress: + - ports: + - port: 9090 +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pdb.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pdb.yaml new file mode 100644 index 000000000..7ffe67307 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pdb.yaml @@ -0,0 +1,15 @@ +{{- if .Values.server.podDisruptionBudget.enabled }} +{{- $pdbSpec := omit .Values.server.podDisruptionBudget "enabled" }} +apiVersion: {{ template "prometheus.podDisruptionBudget.apiVersion" . }} +kind: PodDisruptionBudget +metadata: + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + {{- toYaml $pdbSpec | nindent 2 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/psp.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/psp.yaml new file mode 100644 index 000000000..5776e2541 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/psp.yaml @@ -0,0 +1,53 @@ +{{- if and .Values.rbac.create .Values.podSecurityPolicy.enabled }} +{{- if .Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy" }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ template "prometheus.server.fullname" . }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + {{- with .Values.server.podSecurityPolicy.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + allowedCapabilities: + - 'CHOWN' + volumes: + - 'configMap' + - 'persistentVolumeClaim' + - 'emptyDir' + - 'secret' + - 'hostPath' + allowedHostPaths: + - pathPrefix: /etc + readOnly: true + - pathPrefix: {{ .Values.server.persistentVolume.mountPath }} + {{- range .Values.server.extraHostPathMounts }} + - pathPrefix: {{ .hostPath }} + readOnly: {{ .readOnly }} + {{- end }} + hostNetwork: false + hostPID: false + hostIPC: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pvc.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pvc.yaml new file mode 100644 index 000000000..a9dc4fce0 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/pvc.yaml @@ -0,0 +1,43 @@ +{{- if not .Values.server.statefulSet.enabled -}} +{{- if .Values.server.persistentVolume.enabled -}} +{{- if not .Values.server.persistentVolume.existingClaim -}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + {{- if .Values.server.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.server.persistentVolume.annotations | indent 4 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + {{- with .Values.server.persistentVolume.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +spec: + accessModes: +{{ toYaml .Values.server.persistentVolume.accessModes | indent 4 }} +{{- if .Values.server.persistentVolume.storageClass }} +{{- if (eq "-" .Values.server.persistentVolume.storageClass) }} + storageClassName: "" +{{- else }} + storageClassName: "{{ .Values.server.persistentVolume.storageClass }}" +{{- end }} +{{- end }} +{{- if .Values.server.persistentVolume.volumeBindingMode }} + volumeBindingMode: "{{ .Values.server.persistentVolume.volumeBindingMode }}" +{{- end }} + resources: + requests: + storage: "{{ .Values.server.persistentVolume.size }}" +{{- if .Values.server.persistentVolume.selector }} + selector: + {{- toYaml .Values.server.persistentVolume.selector | nindent 4 }} +{{- end -}} +{{- if .Values.server.persistentVolume.volumeName }} + volumeName: "{{ .Values.server.persistentVolume.volumeName }}" +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/rolebinding.yaml new file mode 100644 index 000000000..721b38816 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/rolebinding.yaml @@ -0,0 +1,18 @@ +{{- range include "prometheus.namespaces" . | fromJsonArray }} +--- +apiVersion: {{ template "rbac.apiVersion" $ }} +kind: RoleBinding +metadata: + labels: + {{- include "prometheus.server.labels" $ | nindent 4 }} + name: {{ template "prometheus.server.fullname" $ }} + namespace: {{ . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.server" $ }} + namespace: {{ include "prometheus.namespace" $ }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ $.Values.server.useExistingClusterRoleName }} +{{ end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/service.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/service.yaml new file mode 100644 index 000000000..069f3270d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/service.yaml @@ -0,0 +1,63 @@ +{{- if .Values.server.service.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.server.service.annotations }} + annotations: +{{ toYaml .Values.server.service.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- if .Values.server.service.labels }} +{{ toYaml .Values.server.service.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +spec: +{{- if .Values.server.service.clusterIP }} + clusterIP: {{ .Values.server.service.clusterIP }} +{{- end }} +{{- if .Values.server.service.externalIPs }} + externalIPs: +{{ toYaml .Values.server.service.externalIPs | indent 4 }} +{{- end }} +{{- if .Values.server.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.server.service.loadBalancerIP }} +{{- end }} +{{- if .Values.server.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.server.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - name: http + port: {{ .Values.server.service.servicePort }} + protocol: TCP + targetPort: 9090 + {{- if .Values.server.service.nodePort }} + nodePort: {{ .Values.server.service.nodePort }} + {{- end }} + {{- if .Values.server.service.gRPC.enabled }} + - name: grpc + port: {{ .Values.server.service.gRPC.servicePort }} + protocol: TCP + targetPort: 10901 + {{- if .Values.server.service.gRPC.nodePort }} + nodePort: {{ .Values.server.service.gRPC.nodePort }} + {{- end }} + {{- end }} +{{- if .Values.server.service.additionalPorts }} +{{ toYaml .Values.server.service.additionalPorts | indent 4 }} +{{- end }} + selector: + {{- if and .Values.server.statefulSet.enabled .Values.server.service.statefulsetReplica.enabled }} + statefulset.kubernetes.io/pod-name: {{ template "prometheus.server.fullname" . }}-{{ .Values.server.service.statefulsetReplica.replica }} + {{- else -}} + {{- include "prometheus.server.matchLabels" . | nindent 4 }} +{{- if .Values.server.service.sessionAffinity }} + sessionAffinity: {{ .Values.server.service.sessionAffinity }} +{{- end }} + {{- end }} + type: "{{ .Values.server.service.type }}" +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/serviceaccount.yaml new file mode 100644 index 000000000..6d5ab0c7d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/serviceaccount.yaml @@ -0,0 +1,16 @@ +{{- if .Values.serviceAccounts.server.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.serviceAccountName.server" . }} + namespace: {{ include "prometheus.namespace" . }} + annotations: +{{ toYaml .Values.serviceAccounts.server.annotations | indent 4 }} +{{- if kindIs "bool" .Values.server.automountServiceAccountToken }} +automountServiceAccountToken: {{ .Values.server.automountServiceAccountToken }} +{{- else if kindIs "bool" .Values.serviceAccounts.server.automountServiceAccountToken }} +automountServiceAccountToken: {{ .Values.serviceAccounts.server.automountServiceAccountToken }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/sts.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/sts.yaml new file mode 100644 index 000000000..6200555df --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/sts.yaml @@ -0,0 +1,436 @@ +{{- if .Values.server.statefulSet.enabled -}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: +{{- if .Values.server.statefulSet.annotations }} + annotations: + {{ toYaml .Values.server.statefulSet.annotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + {{- if .Values.server.statefulSet.labels}} + {{ toYaml .Values.server.statefulSet.labels | nindent 4 }} + {{- end}} + name: {{ template "prometheus.server.fullname" . }} + namespace: {{ include "prometheus.namespace" . }} +spec: + {{- if semverCompare ">= 1.27.x" (include "prometheus.kubeVersion" .) }} + persistentVolumeClaimRetentionPolicy: + whenDeleted: {{ ternary "Delete" "Retain" .Values.server.statefulSet.pvcDeleteOnStsDelete }} + whenScaled: {{ ternary "Delete" "Retain" .Values.server.statefulSet.pvcDeleteOnStsScale }} + {{- end }} + serviceName: {{ template "prometheus.server.fullname" . }}-headless + selector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + replicas: {{ .Values.server.replicaCount }} + revisionHistoryLimit: {{ .Values.server.revisionHistoryLimit }} + podManagementPolicy: {{ .Values.server.statefulSet.podManagementPolicy }} + template: + metadata: + {{- if .Values.server.podAnnotations }} + annotations: + {{ toYaml .Values.server.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 8 }} + {{- if .Values.server.podLabels}} + {{ toYaml .Values.server.podLabels | nindent 8 }} + {{- end}} + spec: +{{- if .Values.server.priorityClassName }} + priorityClassName: "{{ .Values.server.priorityClassName }}" +{{- end }} +{{- if .Values.server.schedulerName }} + schedulerName: "{{ .Values.server.schedulerName }}" +{{- end }} +{{- if semverCompare ">=1.13-0" .Capabilities.KubeVersion.GitVersion }} + {{- if or (.Values.server.enableServiceLinks) (eq (.Values.server.enableServiceLinks | toString) "") }} + enableServiceLinks: true + {{- else }} + enableServiceLinks: false + {{- end }} +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.server" . }} +{{- if kindIs "bool" .Values.server.automountServiceAccountToken }} + automountServiceAccountToken: {{ .Values.server.automountServiceAccountToken }} +{{- end }} + {{- if .Values.server.extraInitContainers }} + initContainers: +{{ toYaml .Values.server.extraInitContainers | indent 8 }} + {{- end }} + containers: + {{- if .Values.configmapReload.prometheus.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }}-{{ .Values.configmapReload.prometheus.name }} + {{- if .Values.configmapReload.prometheus.image.digest }} + image: "{{ .Values.configmapReload.prometheus.image.repository }}@{{ .Values.configmapReload.prometheus.image.digest }}" + {{- else }} + image: "{{ .Values.configmapReload.prometheus.image.repository }}:{{ .Values.configmapReload.prometheus.image.tag }}" + {{- end }} + imagePullPolicy: "{{ .Values.configmapReload.prometheus.image.pullPolicy }}" + {{- with .Values.configmapReload.prometheus.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + args: + - --watched-dir=/etc/config + {{- $default_url := "http://127.0.0.1:9090/-/reload" }} + {{- with .Values.server.prefixURL }} + {{- $default_url = printf "http://127.0.0.1:9090%s/-/reload" . }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + - --listen-address=0.0.0.0:{{ .Values.configmapReload.prometheus.containerPort }} + {{- end }} + - --reload-url={{ default $default_url .Values.configmapReload.reloadUrl }} + {{- range $key, $value := .Values.configmapReload.prometheus.extraArgs }} + {{- if $value }} + - --{{ $key }}={{ $value }} + {{- else }} + - --{{ $key }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraVolumeDirs }} + - --watched-dir={{ . }} + {{- end }} + {{- with .Values.configmapReload.env }} + env: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + ports: + - containerPort: {{ .Values.configmapReload.prometheus.containerPort }} + {{- if .Values.configmapReload.prometheus.containerPortName }} + name: {{ .Values.configmapReload.prometheus.containerPortName }} + {{- end }} + {{- end }} + {{- with .Values.configmapReload.prometheus.livenessProbe }} + livenessProbe: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.configmapReload.prometheus.readinessProbe }} + readinessProbe: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.configmapReload.prometheus.startupProbe }} + {{- $startupProbe := omit .Values.configmapReload.prometheus.startupProbe "enabled" }} + startupProbe: + {{- toYaml $startupProbe | nindent 12 }} + {{- end }} + {{- with .Values.configmapReload.prometheus.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- with .Values.configmapReload.prometheus.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- end }} + + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }} + {{- if .Values.server.image.digest }} + image: "{{ .Values.server.image.repository }}@{{ .Values.server.image.digest }}" + {{- else }} + image: "{{ .Values.server.image.repository }}:{{ .Values.server.image.tag | default .Chart.AppVersion }}" + {{- end }} + imagePullPolicy: "{{ .Values.server.image.pullPolicy }}" + {{- with .Values.server.command }} + command: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.server.env }} + env: +{{ toYaml .Values.server.env | indent 12}} + {{- end }} + args: + {{- if .Values.server.defaultFlagsOverride }} + {{ toYaml .Values.server.defaultFlagsOverride | nindent 12}} + {{- else }} + {{- if .Values.server.prefixURL }} + - --web.route-prefix={{ .Values.server.prefixURL }} + {{- end }} + {{- if .Values.server.retention }} + - --storage.tsdb.retention.time={{ .Values.server.retention }} + {{- end }} + {{- if .Values.server.retentionSize }} + - --storage.tsdb.retention.size={{ .Values.server.retentionSize }} + {{- end }} + - --config.file={{ .Values.server.configPath }} + {{- if .Values.server.storagePath }} + - --storage.tsdb.path={{ .Values.server.storagePath }} + {{- else }} + - --storage.tsdb.path={{ .Values.server.persistentVolume.mountPath }} + {{- end }} + - --web.console.libraries=/etc/prometheus/console_libraries + - --web.console.templates=/etc/prometheus/consoles + {{- range .Values.server.extraFlags }} + - --{{ . }} + {{- end }} + {{- range $key, $value := .Values.server.extraArgs }} + {{- if $value }} + - --{{ $key }}={{ $value }} + {{- else }} + - --{{ $key }} + {{- end }} + {{- end }} + {{- if .Values.server.baseURL }} + - --web.external-url={{ .Values.server.baseURL }} + {{- end }} + {{- end }} + ports: + - containerPort: 9090 + {{- if .Values.server.portName }} + name: {{ .Values.server.portName }} + {{- end }} + {{- if .Values.server.hostPort }} + hostPort: {{ .Values.server.hostPort }} + {{- end }} + readinessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/ready + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- with .Values.server.probeHeaders }} + httpHeaders: +{{- toYaml . | nindent 14 }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.readinessProbeInitialDelay }} + periodSeconds: {{ .Values.server.readinessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.readinessProbeTimeout }} + failureThreshold: {{ .Values.server.readinessProbeFailureThreshold }} + successThreshold: {{ .Values.server.readinessProbeSuccessThreshold }} + livenessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- with .Values.server.probeHeaders }} + httpHeaders: +{{- toYaml . | nindent 14 }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.livenessProbeInitialDelay }} + periodSeconds: {{ .Values.server.livenessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.livenessProbeTimeout }} + failureThreshold: {{ .Values.server.livenessProbeFailureThreshold }} + successThreshold: {{ .Values.server.livenessProbeSuccessThreshold }} + {{- if .Values.server.startupProbe.enabled }} + startupProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- if .Values.server.probeHeaders }} + httpHeaders: + {{- range .Values.server.probeHeaders}} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + failureThreshold: {{ .Values.server.startupProbe.failureThreshold }} + periodSeconds: {{ .Values.server.startupProbe.periodSeconds }} + timeoutSeconds: {{ .Values.server.startupProbe.timeoutSeconds }} + {{- end }} + {{- with .Values.server.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: {{ ternary .Values.server.persistentVolume.statefulSetNameOverride "storage-volume" (and .Values.server.persistentVolume.enabled (not (empty .Values.server.persistentVolume.statefulSetNameOverride))) }} + mountPath: {{ .Values.server.persistentVolume.mountPath }} + subPath: "{{ .Values.server.persistentVolume.subPath }}" + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.server.extraVolumeMounts }} + {{ toYaml .Values.server.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- with .Values.server.containerSecurityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- if .Values.server.sidecarContainers }} + {{- range $name, $spec := .Values.server.sidecarContainers }} + - name: {{ $name }} + {{- if kindIs "string" $spec }} + {{- tpl $spec $ | nindent 10 }} + {{- else }} + {{- toYaml $spec | nindent 10 }} + {{- end }} + {{- end }} + {{- end }} + hostNetwork: {{ .Values.server.hostNetwork }} + {{- if .Values.server.dnsPolicy }} + dnsPolicy: {{ .Values.server.dnsPolicy }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.server.nodeSelector }} + nodeSelector: +{{ toYaml .Values.server.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.server.hostAliases }} + hostAliases: +{{ toYaml .Values.server.hostAliases | indent 8 }} + {{- end }} + {{- if .Values.server.dnsConfig }} + dnsConfig: +{{ toYaml .Values.server.dnsConfig | indent 8 }} + {{- end }} + {{- with .Values.server.securityContext }} + securityContext: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.server.tolerations }} + tolerations: +{{ toYaml .Values.server.tolerations | indent 8 }} + {{- end }} + {{- if or .Values.server.affinity .Values.server.podAntiAffinity }} + affinity: + {{- end }} + {{- with .Values.server.affinity }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if eq .Values.server.podAntiAffinity "hard" }} + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - topologyKey: {{ .Values.server.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ template "prometheus.name" . }}]} + {{- else if eq .Values.server.podAntiAffinity "soft" }} + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + topologyKey: {{ .Values.server.podAntiAffinityTopologyKey }} + labelSelector: + matchExpressions: + - {key: app.kubernetes.io/name, operator: In, values: [{{ template "prometheus.name" . }}]} + {{- end }} + {{- with .Values.server.topologySpreadConstraints }} + topologySpreadConstraints: + {{- toYaml . | nindent 8 }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.server.terminationGracePeriodSeconds }} + volumes: + - name: config-volume + {{- if empty .Values.server.configFromSecret }} + configMap: + name: {{ if .Values.server.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.server.configMapOverrideName }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.server.configFromSecret }} + {{- end }} + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + hostPath: + path: {{ .hostPath }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} +{{- if .Values.server.extraVolumes }} +{{ toYaml .Values.server.extraVolumes | indent 8}} +{{- end }} +{{- if .Values.server.persistentVolume.enabled }} + volumeClaimTemplates: + - apiVersion: v1 + kind: PersistentVolumeClaim + metadata: + name: {{ .Values.server.persistentVolume.statefulSetNameOverride | default "storage-volume" }} + {{- if .Values.server.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.server.persistentVolume.annotations | indent 10 }} + {{- end }} + {{- if .Values.server.persistentVolume.labels }} + labels: +{{ toYaml .Values.server.persistentVolume.labels | indent 10 }} + {{- end }} + spec: + accessModes: +{{ toYaml .Values.server.persistentVolume.accessModes | indent 10 }} + resources: + requests: + storage: "{{ .Values.server.persistentVolume.size }}" + {{- if .Values.server.persistentVolume.storageClass }} + {{- if (eq "-" .Values.server.persistentVolume.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.server.persistentVolume.storageClass }}" + {{- end }} + {{- end }} +{{- else }} + - name: storage-volume + emptyDir: + {{- if .Values.server.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.server.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/templates/vpa.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/vpa.yaml new file mode 100644 index 000000000..cd07ad8b7 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/templates/vpa.yaml @@ -0,0 +1,26 @@ +{{- if .Values.server.verticalAutoscaler.enabled -}} +{{- if .Capabilities.APIVersions.Has "autoscaling.k8s.io/v1/VerticalPodAutoscaler" }} +apiVersion: autoscaling.k8s.io/v1 +{{- else }} +apiVersion: autoscaling.k8s.io/v1beta2 +{{- end }} +kind: VerticalPodAutoscaler +metadata: + name: {{ template "prometheus.server.fullname" . }}-vpa + namespace: {{ include "prometheus.namespace" . }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +spec: + targetRef: + apiVersion: "apps/v1" +{{- if .Values.server.statefulSet.enabled }} + kind: StatefulSet +{{- else }} + kind: Deployment +{{- end }} + name: {{ template "prometheus.server.fullname" . }} + updatePolicy: + updateMode: {{ .Values.server.verticalAutoscaler.updateMode | default "Off" | quote }} + resourcePolicy: + containerPolicies: {{ .Values.server.verticalAutoscaler.containerPolicies | default list | toYaml | trim | nindent 4 }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/values.schema.json b/charts/kasten/k10/7.0.1401/charts/prometheus/values.schema.json new file mode 100644 index 000000000..b2d8af26c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/values.schema.json @@ -0,0 +1,752 @@ +{ + "$schema": "http://json-schema.org/schema#", + "type": "object", + "properties": { + "alertRelabelConfigs": { + "type": "object" + }, + "alertmanager": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "persistence": { + "type": "object", + "properties": { + "size": { + "type": "string" + } + } + }, + "podSecurityContext": { + "type": "object", + "properties": { + "fsGroup": { + "type": "integer" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + } + } + } + } + }, + "configmapReload": { + "type": "object", + "properties": { + "env": { + "type": "array" + }, + "prometheus": { + "type": "object", + "properties": { + "containerSecurityContext": { + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "extraArgs": { + "type": "object" + }, + "extraConfigmapMounts": { + "type": "array" + }, + "extraVolumeDirs": { + "type": "array" + }, + "extraVolumeMounts": { + "type": "array" + }, + "image": { + "type": "object", + "properties": { + "digest": { + "type": "string" + }, + "pullPolicy": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "tag": { + "type": "string" + } + } + }, + "name": { + "type": "string" + }, + "resources": { + "type": "object" + } + } + }, + "reloadUrl": { + "type": "string" + } + } + }, + "extraManifests": { + "type": "array" + }, + "extraScrapeConfigs": { + "type": "string" + }, + "forceNamespace": { + "type": "string" + }, + "imagePullSecrets": { + "type": "array" + }, + "kube-state-metrics": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + } + }, + "networkPolicy": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + } + }, + "podSecurityPolicy": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + } + }, + "prometheus-node-exporter": { + "type": "object", + "properties": { + "containerSecurityContext": { + "type": "object", + "properties": { + "allowPrivilegeEscalation": { + "type": "boolean" + } + } + }, + "enabled": { + "type": "boolean" + }, + "rbac": { + "type": "object", + "properties": { + "pspEnabled": { + "type": "boolean" + } + } + } + } + }, + "prometheus-pushgateway": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "serviceAnnotations": { + "type": "object", + "properties": { + "prometheus.io/probe": { + "type": "string" + } + } + } + } + }, + "rbac": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + } + } + }, + "ruleFiles": { + "type": "object" + }, + "server": { + "type": "object", + "properties": { + "affinity": { + "type": "object" + }, + "alertmanagers": { + "type": "array" + }, + "baseURL": { + "type": "string" + }, + "clusterRoleNameOverride": { + "type": "string" + }, + "command": { + "type": "array" + }, + "configMapAnnotations": { + "type": "object" + }, + "configMapOverrideName": { + "type": "string" + }, + "configPath": { + "type": "string" + }, + "containerSecurityContext": { + "type": "object" + }, + "defaultFlagsOverride": { + "type": "array" + }, + "deploymentAnnotations": { + "type": "object" + }, + "dnsConfig": { + "type": "object" + }, + "dnsPolicy": { + "type": "string" + }, + "emptyDir": { + "type": "object", + "properties": { + "sizeLimit": { + "type": "string" + } + } + }, + "enableServiceLinks": { + "type": "boolean" + }, + "env": { + "type": "array" + }, + "exemplars": { + "type": "object" + }, + "extraArgs": { + "type": "object" + }, + "extraConfigmapLabels": { + "type": "object" + }, + "extraConfigmapMounts": { + "type": "array" + }, + "extraFlags": { + "type": "array", + "items": { + "type": "string" + } + }, + "extraHostPathMounts": { + "type": "array" + }, + "extraInitContainers": { + "type": "array" + }, + "extraSecretMounts": { + "type": "array" + }, + "extraVolumeMounts": { + "type": "array" + }, + "extraVolumes": { + "type": "array" + }, + "fullnameOverride": { + "type": "string" + }, + "global": { + "type": "object", + "properties": { + "evaluation_interval": { + "type": "string" + }, + "scrape_interval": { + "type": "string" + }, + "scrape_timeout": { + "type": "string" + } + } + }, + "hostAliases": { + "type": "array" + }, + "hostNetwork": { + "type": "boolean" + }, + "image": { + "type": "object", + "properties": { + "digest": { + "type": "string" + }, + "pullPolicy": { + "type": "string" + }, + "repository": { + "type": "string" + }, + "tag": { + "type": "string" + } + } + }, + "ingress": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "extraLabels": { + "type": "object" + }, + "extraPaths": { + "type": "array" + }, + "hosts": { + "type": "array" + }, + "path": { + "type": "string" + }, + "pathType": { + "type": "string" + }, + "tls": { + "type": "array" + } + } + }, + "livenessProbeFailureThreshold": { + "type": "integer" + }, + "livenessProbeInitialDelay": { + "type": "integer" + }, + "livenessProbePeriodSeconds": { + "type": "integer" + }, + "livenessProbeSuccessThreshold": { + "type": "integer" + }, + "livenessProbeTimeout": { + "type": "integer" + }, + "name": { + "type": "string" + }, + "nodeSelector": { + "type": "object" + }, + "persistentVolume": { + "type": "object", + "properties": { + "accessModes": { + "type": "array", + "items": { + "type": "string" + } + }, + "annotations": { + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "existingClaim": { + "type": "string" + }, + "labels": { + "type": "object" + }, + "mountPath": { + "type": "string" + }, + "size": { + "type": "string" + }, + "statefulSetNameOverride": { + "type": "string" + }, + "subPath": { + "type": "string" + } + } + }, + "podAnnotations": { + "type": "object" + }, + "podAntiAffinity": { + "type": "string", + "enum": ["", "soft", "hard"], + "default": "" + }, + "podAntiAffinityTopologyKey": { + "type": "string" + }, + "podDisruptionBudget": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "maxUnavailable": { + "type": [ + "string", + "integer" + ] + } + } + }, + "podLabels": { + "type": "object" + }, + "podSecurityPolicy": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + } + } + }, + "portName": { + "type": "string" + }, + "prefixURL": { + "type": "string" + }, + "priorityClassName": { + "type": "string" + }, + "probeHeaders": { + "type": "array" + }, + "probeScheme": { + "type": "string" + }, + "readinessProbeFailureThreshold": { + "type": "integer" + }, + "readinessProbeInitialDelay": { + "type": "integer" + }, + "readinessProbePeriodSeconds": { + "type": "integer" + }, + "readinessProbeSuccessThreshold": { + "type": "integer" + }, + "readinessProbeTimeout": { + "type": "integer" + }, + "releaseNamespace": { + "type": "boolean" + }, + "remoteRead": { + "type": "array" + }, + "remoteWrite": { + "type": "array" + }, + "replicaCount": { + "type": "integer" + }, + "resources": { + "type": "object" + }, + "retention": { + "type": "string" + }, + "retentionSize": { + "type": "string" + }, + "revisionHistoryLimit": { + "type": "integer" + }, + "securityContext": { + "type": "object", + "properties": { + "fsGroup": { + "type": "integer" + }, + "runAsGroup": { + "type": "integer" + }, + "runAsNonRoot": { + "type": "boolean" + }, + "runAsUser": { + "type": "integer" + } + } + }, + "service": { + "type": "object", + "properties": { + "additionalPorts": { + "type": "array" + }, + "annotations": { + "type": "object" + }, + "clusterIP": { + "type": "string" + }, + "enabled": { + "type": "boolean" + }, + "externalIPs": { + "type": "array" + }, + "gRPC": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "servicePort": { + "type": "integer" + } + } + }, + "labels": { + "type": "object" + }, + "loadBalancerIP": { + "type": "string" + }, + "loadBalancerSourceRanges": { + "type": "array" + }, + "servicePort": { + "type": "integer" + }, + "sessionAffinity": { + "type": "string" + }, + "statefulsetReplica": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "replica": { + "type": "integer" + } + } + }, + "type": { + "type": "string" + } + } + }, + "sidecarContainers": { + "type": "object" + }, + "sidecarTemplateValues": { + "type": "object" + }, + "startupProbe": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "failureThreshold": { + "type": "integer" + }, + "periodSeconds": { + "type": "integer" + }, + "timeoutSeconds": { + "type": "integer" + } + } + }, + "statefulSet": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "enabled": { + "type": "boolean" + }, + "headless": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "gRPC": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + }, + "servicePort": { + "type": "integer" + } + } + }, + "labels": { + "type": "object" + }, + "servicePort": { + "type": "integer" + } + } + }, + "labels": { + "type": "object" + }, + "podManagementPolicy": { + "type": "string" + }, + "pvcDeleteOnStsDelete": { + "type": "boolean" + }, + "pvcDeleteOnStsScale": { + "type": "boolean" + } + } + }, + "storagePath": { + "type": "string" + }, + "strategy": { + "type": "object", + "properties": { + "type": { + "type": "string" + } + } + }, + "tcpSocketProbeEnabled": { + "type": "boolean" + }, + "terminationGracePeriodSeconds": { + "type": "integer" + }, + "tolerations": { + "type": "array" + }, + "topologySpreadConstraints": { + "type": "array" + }, + "tsdb": { + "type": "object" + }, + "verticalAutoscaler": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + } + } + } + }, + "scrapeConfigFiles": { + "type": "array" + }, + "serverFiles": { + "type": "object", + "properties": { + "alerting_rules.yml": { + "type": "object" + }, + "alerts": { + "type": "object" + }, + "prometheus.yml": { + "type": "object", + "properties": { + "rule_files": { + "type": "array", + "items": { + "type": "string" + } + }, + "scrape_configs": { + "type": "array", + "items": { + "type": "object", + "properties": { + "job_name": { + "type": "string" + }, + "static_configs": { + "type": "array", + "items": { + "type": "object", + "properties": { + "targets": { + "type": "array", + "items": { + "type": "string" + } + } + } + } + } + } + } + } + } + }, + "recording_rules.yml": { + "type": "object" + }, + "rules": { + "type": "object" + } + } + }, + "serviceAccounts": { + "type": "object", + "properties": { + "server": { + "type": "object", + "properties": { + "annotations": { + "type": "object" + }, + "create": { + "type": "boolean" + }, + "name": { + "type": "string" + }, + "automountServiceAccountToken": { + "type": "boolean" + } + } + } + } + } + } +} diff --git a/charts/kasten/k10/7.0.1401/charts/prometheus/values.yaml b/charts/kasten/k10/7.0.1401/charts/prometheus/values.yaml new file mode 100644 index 000000000..6d3733f56 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/charts/prometheus/values.yaml @@ -0,0 +1,1315 @@ +# yaml-language-server: $schema=values.schema.json +# Default values for prometheus. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +rbac: + create: true + +podSecurityPolicy: + enabled: false + +imagePullSecrets: [] +# - name: "image-pull-secret" + +## Define serviceAccount names for components. Defaults to component's fully qualified name. +## +serviceAccounts: + server: + create: true + name: "" + annotations: {} + + ## Opt out of automounting Kubernetes API credentials. + ## It will be overriden by server.automountServiceAccountToken value, if set. + # automountServiceAccountToken: false + +## Additional labels to attach to all resources +commonMetaLabels: {} + +## Monitors ConfigMap changes and POSTs to a URL +## Ref: https://github.com/prometheus-operator/prometheus-operator/tree/main/cmd/prometheus-config-reloader +## +configmapReload: + ## URL for configmap-reload to use for reloads + ## + reloadUrl: "" + + ## env sets environment variables to pass to the container. Can be set as name/value pairs, + ## read from secrets or configmaps. + env: [] + # - name: SOMEVAR + # value: somevalue + # - name: PASSWORD + # valueFrom: + # secretKeyRef: + # name: mysecret + # key: password + # optional: false + + prometheus: + ## If false, the configmap-reload container will not be deployed + ## + enabled: true + + ## configmap-reload container name + ## + name: configmap-reload + + ## configmap-reload container image + ## + image: + repository: quay.io/prometheus-operator/prometheus-config-reloader + tag: v0.76.2 + # When digest is set to a non-empty value, images will be pulled by digest (regardless of tag value). + digest: "" + pullPolicy: IfNotPresent + + ## config-reloader's container port and port name for probes and metrics + containerPort: 8080 + containerPortName: metrics + + ## Additional configmap-reload container arguments + ## Set to null for argumentless flags + ## + extraArgs: {} + + ## Additional configmap-reload volume directories + ## + extraVolumeDirs: [] + + ## Additional configmap-reload volume mounts + ## + extraVolumeMounts: [] + + ## Additional configmap-reload mounts + ## + extraConfigmapMounts: [] + # - name: prometheus-alerts + # mountPath: /etc/alerts.d + # subPath: "" + # configMap: prometheus-alerts + # readOnly: true + + ## Security context to be added to configmap-reload container + containerSecurityContext: {} + + ## Settings for Prometheus reloader's readiness, liveness and startup probes + ## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/ + ## + + livenessProbe: + httpGet: + path: /healthz + port: metrics + scheme: HTTP + periodSeconds: 10 + initialDelaySeconds: 2 + + readinessProbe: + httpGet: + path: /healthz + port: metrics + scheme: HTTP + periodSeconds: 10 + + startupProbe: + enabled: false + httpGet: + path: /healthz + port: metrics + scheme: HTTP + periodSeconds: 10 + + ## configmap-reload resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + requests: + cpu: 10m + memory: 50Mi + +server: + ## Prometheus server container name + ## + name: server + + ## Opt out of automounting Kubernetes API credentials. + ## If set it will override serviceAccounts.server.automountServiceAccountToken value for ServiceAccount. + # automountServiceAccountToken: false + + ## Use a ClusterRole (and ClusterRoleBinding) + ## - If set to false - we define a RoleBinding in the defined namespaces ONLY + ## + ## NB: because we need a Role with nonResourceURL's ("/metrics") - you must get someone with Cluster-admin privileges to define this role for you, before running with this setting enabled. + ## This makes prometheus work - for users who do not have ClusterAdmin privs, but wants prometheus to operate on their own namespaces, instead of clusterwide. + ## + ## You MUST also set namespaces to the ones you have access to and want monitored by Prometheus. + ## + # useExistingClusterRoleName: nameofclusterrole + + ## If set it will override prometheus.server.fullname value for ClusterRole and ClusterRoleBinding + ## + clusterRoleNameOverride: "" + + # Enable only the release namespace for monitoring. By default all namespaces are monitored. + # If releaseNamespace and namespaces are both set a merged list will be monitored. + releaseNamespace: false + + ## namespaces to monitor (instead of monitoring all - clusterwide). Needed if you want to run without Cluster-admin privileges. + # namespaces: + # - yournamespace + + # sidecarContainers - add more containers to prometheus server + # Key/Value where Key is the sidecar `- name: ` + # Example: + # sidecarContainers: + # webserver: + # image: nginx + # OR for adding OAuth authentication to Prometheus + # sidecarContainers: + # oauth-proxy: + # image: quay.io/oauth2-proxy/oauth2-proxy:v7.1.2 + # args: + # - --upstream=http://127.0.0.1:9090 + # - --http-address=0.0.0.0:8081 + # - ... + # ports: + # - containerPort: 8081 + # name: oauth-proxy + # protocol: TCP + # resources: {} + sidecarContainers: {} + + # sidecarTemplateValues - context to be used in template for sidecarContainers + # Example: + # sidecarTemplateValues: *your-custom-globals + # sidecarContainers: + # webserver: |- + # {{ include "webserver-container-template" . }} + # Template for `webserver-container-template` might looks like this: + # image: "{{ .Values.server.sidecarTemplateValues.repository }}:{{ .Values.server.sidecarTemplateValues.tag }}" + # ... + # + sidecarTemplateValues: {} + + ## Prometheus server container image + ## + image: + repository: quay.io/prometheus/prometheus + # if not set appVersion field from Chart.yaml is used + tag: "" + # When digest is set to a non-empty value, images will be pulled by digest (regardless of tag value). + digest: "" + pullPolicy: IfNotPresent + + ## Prometheus server command + ## + command: [] + + ## prometheus server priorityClassName + ## + priorityClassName: "" + + ## EnableServiceLinks indicates whether information about services should be injected + ## into pod's environment variables, matching the syntax of Docker links. + ## WARNING: the field is unsupported and will be skipped in K8s prior to v1.13.0. + ## + enableServiceLinks: true + + ## The URL prefix at which the container can be accessed. Useful in the case the '-web.external-url' includes a slug + ## so that the various internal URLs are still able to access as they are in the default case. + ## (Optional) + prefixURL: "" + + ## External URL which can access prometheus + ## Maybe same with Ingress host name + baseURL: "" + + ## Additional server container environment variables + ## + ## You specify this manually like you would a raw deployment manifest. + ## This means you can bind in environment variables from secrets. + ## + ## e.g. static environment variable: + ## - name: DEMO_GREETING + ## value: "Hello from the environment" + ## + ## e.g. secret environment variable: + ## - name: USERNAME + ## valueFrom: + ## secretKeyRef: + ## name: mysecret + ## key: username + env: [] + + # List of flags to override default parameters, e.g: + # - --enable-feature=agent + # - --storage.agent.retention.max-time=30m + # - --config.file=/etc/config/prometheus.yml + defaultFlagsOverride: [] + + extraFlags: + - web.enable-lifecycle + ## web.enable-admin-api flag controls access to the administrative HTTP API which includes functionality such as + ## deleting time series. This is disabled by default. + # - web.enable-admin-api + ## + ## storage.tsdb.no-lockfile flag controls BD locking + # - storage.tsdb.no-lockfile + ## + ## storage.tsdb.wal-compression flag enables compression of the write-ahead log (WAL) + # - storage.tsdb.wal-compression + + ## Path to a configuration file on prometheus server container FS + configPath: /etc/config/prometheus.yml + + ### The data directory used by prometheus to set --storage.tsdb.path + ### When empty server.persistentVolume.mountPath is used instead + storagePath: "" + + global: + ## How frequently to scrape targets by default + ## + scrape_interval: 1m + ## How long until a scrape request times out + ## + scrape_timeout: 10s + ## How frequently to evaluate rules + ## + evaluation_interval: 1m + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write + ## + remoteWrite: [] + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_read + ## + remoteRead: [] + + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#tsdb + ## + tsdb: {} + # out_of_order_time_window: 0s + + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#exemplars + ## Must be enabled via --enable-feature=exemplar-storage + ## + exemplars: {} + # max_exemplars: 100000 + + ## Custom HTTP headers for Liveness/Readiness/Startup Probe + ## + ## Useful for providing HTTP Basic Auth to healthchecks + probeHeaders: [] + # - name: "Authorization" + # value: "Bearer ABCDEabcde12345" + + ## Additional Prometheus server container arguments + ## Set to null for argumentless flags + ## + extraArgs: {} + # web.enable-remote-write-receiver: null + + ## Additional InitContainers to initialize the pod + ## + extraInitContainers: [] + + ## Additional Prometheus server Volume mounts + ## + extraVolumeMounts: [] + + ## Additional Prometheus server Volumes + ## + extraVolumes: [] + + ## Additional Prometheus server hostPath mounts + ## + extraHostPathMounts: [] + # - name: certs-dir + # mountPath: /etc/kubernetes/certs + # subPath: "" + # hostPath: /etc/kubernetes/certs + # readOnly: true + + extraConfigmapMounts: [] + # - name: certs-configmap + # mountPath: /prometheus + # subPath: "" + # configMap: certs-configmap + # readOnly: true + + ## Additional Prometheus server Secret mounts + # Defines additional mounts with secrets. Secrets must be manually created in the namespace. + extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # subPath: "" + # secretName: prom-secret-files + # readOnly: true + + ## ConfigMap override where fullname is {{.Release.Name}}-{{.Values.server.configMapOverrideName}} + ## Defining configMapOverrideName will cause templates/server-configmap.yaml + ## to NOT generate a ConfigMap resource + ## + configMapOverrideName: "" + + ## Extra labels for Prometheus server ConfigMap (ConfigMap that holds serverFiles) + extraConfigmapLabels: {} + + ## Override the prometheus.server.fullname for all objects related to the Prometheus server + fullnameOverride: "" + + ingress: + ## If true, Prometheus server Ingress will be created + ## + enabled: false + + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + + ## Prometheus server Ingress annotations + ## + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: 'true' + + ## Prometheus server Ingress additional labels + ## + extraLabels: {} + + ## Redirect ingress to an additional defined port on the service + # servicePort: 8081 + + ## Prometheus server Ingress hostnames with optional path + ## Must be provided if Ingress is enabled + ## + hosts: [] + # - prometheus.domain.com + # - domain.com/prometheus + + path: / + + # pathType is only for k8s >= 1.18 + pathType: Prefix + + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + + ## Prometheus server Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: [] + # - secretName: prometheus-server-tls + # hosts: + # - prometheus.domain.com + + ## Server Deployment Strategy type + strategy: + type: Recreate + + ## hostAliases allows adding entries to /etc/hosts inside the containers + hostAliases: [] + # - ip: "127.0.0.1" + # hostnames: + # - "example.com" + + ## Node tolerations for server scheduling to nodes with taints + ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/ + ## + tolerations: [] + # - key: "key" + # operator: "Equal|Exists" + # value: "value" + # effect: "NoSchedule|PreferNoSchedule|NoExecute(1.6 only)" + + ## Node labels for Prometheus server pod assignment + ## Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/ + ## + nodeSelector: {} + + ## Pod affinity + ## + affinity: {} + + ## Pod anti-affinity can prevent the scheduler from placing Prometheus server replicas on the same node. + ## The value "soft" means that the scheduler should *prefer* to not schedule two replica pods onto the same node but no guarantee is provided. + ## The value "hard" means that the scheduler is *required* to not schedule two replica pods onto the same node. + ## The default value "" will disable pod anti-affinity so that no anti-affinity rules will be configured (unless set in `server.affinity`). + ## + podAntiAffinity: "" + + ## If anti-affinity is enabled sets the topologyKey to use for anti-affinity. + ## This can be changed to, for example, failure-domain.beta.kubernetes.io/zone + ## + podAntiAffinityTopologyKey: kubernetes.io/hostname + + ## Pod topology spread constraints + ## ref. https://kubernetes.io/docs/concepts/scheduling-eviction/topology-spread-constraints/ + topologySpreadConstraints: [] + + ## PodDisruptionBudget settings + ## ref: https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ + ## + podDisruptionBudget: + enabled: false + maxUnavailable: 1 + # minAvailable: 1 + ## unhealthyPodEvictionPolicy is available since 1.27.0 (beta) + ## https://kubernetes.io/docs/tasks/run-application/configure-pdb/#unhealthy-pod-eviction-policy + # unhealthyPodEvictionPolicy: IfHealthyBudget + + ## Use an alternate scheduler, e.g. "stork". + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + ## + # schedulerName: + + persistentVolume: + ## If true, Prometheus server will create/use a Persistent Volume Claim + ## If false, use emptyDir + ## + enabled: true + + ## If set it will override the name of the created persistent volume claim + ## generated by the stateful set. + ## + statefulSetNameOverride: "" + + ## Prometheus server data Persistent Volume access modes + ## Must match those of existing PV or dynamic provisioner + ## Ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ + ## + accessModes: + - ReadWriteOnce + + ## Prometheus server data Persistent Volume labels + ## + labels: {} + + ## Prometheus server data Persistent Volume annotations + ## + annotations: {} + + ## Prometheus server data Persistent Volume existing claim name + ## Requires server.persistentVolume.enabled: true + ## If defined, PVC must be created manually before volume will be bound + existingClaim: "" + + ## Prometheus server data Persistent Volume mount root path + ## + mountPath: /data + + ## Prometheus server data Persistent Volume size + ## + size: 8Gi + + ## Prometheus server data Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + # storageClass: "-" + + ## Prometheus server data Persistent Volume Binding Mode + ## If defined, volumeBindingMode: + ## If undefined (the default) or set to null, no volumeBindingMode spec is + ## set, choosing the default mode. + ## + # volumeBindingMode: "" + + ## Subdirectory of Prometheus server data Persistent Volume to mount + ## Useful if the volume's root directory is not empty + ## + subPath: "" + + ## Persistent Volume Claim Selector + ## Useful if Persistent Volumes have been provisioned in advance + ## Ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#selector + ## + # selector: + # matchLabels: + # release: "stable" + # matchExpressions: + # - { key: environment, operator: In, values: [ dev ] } + + ## Persistent Volume Name + ## Useful if Persistent Volumes have been provisioned in advance and you want to use a specific one + ## + # volumeName: "" + + emptyDir: + ## Prometheus server emptyDir volume size limit + ## + sizeLimit: "" + + ## Annotations to be added to Prometheus server pods + ## + podAnnotations: {} + # iam.amazonaws.com/role: prometheus + + ## Labels to be added to Prometheus server pods + ## + podLabels: {} + + ## Prometheus AlertManager configuration + ## + alertmanagers: [] + + ## Specify if a Pod Security Policy for node-exporter must be created + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + ## + podSecurityPolicy: + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + ## Use a StatefulSet if replicaCount needs to be greater than 1 (see below) + ## + replicaCount: 1 + + ## Number of old history to retain to allow rollback + ## Default Kubernetes value is set to 10 + ## + revisionHistoryLimit: 10 + + ## Annotations to be added to ConfigMap + ## + configMapAnnotations: {} + + ## Annotations to be added to deployment + ## + deploymentAnnotations: {} + + statefulSet: + ## If true, use a statefulset instead of a deployment for pod management. + ## This allows to scale replicas to more than 1 pod + ## + enabled: false + + annotations: {} + labels: {} + podManagementPolicy: OrderedReady + + ## Alertmanager headless service to use for the statefulset + ## + headless: + annotations: {} + labels: {} + servicePort: 80 + ## Enable gRPC port on service to allow auto discovery with thanos-querier + gRPC: + enabled: false + servicePort: 10901 + # nodePort: 10901 + + ## Statefulset's persistent volume claim retention policy + ## pvcDeleteOnStsDelete and pvcDeleteOnStsScale determine whether + ## statefulset's PVCs are deleted (true) or retained (false) on scaling down + ## and deleting statefulset, respectively. Requires 1.27.0+. + ## Ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#persistentvolumeclaim-retention + ## + pvcDeleteOnStsDelete: false + pvcDeleteOnStsScale: false + + ## Prometheus server readiness and liveness probe initial delay and timeout + ## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/ + ## + tcpSocketProbeEnabled: false + probeScheme: HTTP + readinessProbeInitialDelay: 30 + readinessProbePeriodSeconds: 5 + readinessProbeTimeout: 4 + readinessProbeFailureThreshold: 3 + readinessProbeSuccessThreshold: 1 + livenessProbeInitialDelay: 30 + livenessProbePeriodSeconds: 15 + livenessProbeTimeout: 10 + livenessProbeFailureThreshold: 3 + livenessProbeSuccessThreshold: 1 + startupProbe: + enabled: false + periodSeconds: 5 + failureThreshold: 30 + timeoutSeconds: 10 + + ## Prometheus server resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + requests: + cpu: 20m + memory: 200Mi + # limits: + # cpu: 500m + # memory: 512Mi + # requests: + # cpu: 500m + # memory: 512Mi + + # Required for use in managed kubernetes clusters (such as AWS EKS) with custom CNI (such as calico), + # because control-plane managed by AWS cannot communicate with pods' IP CIDR and admission webhooks are not working + ## + hostNetwork: false + + # When hostNetwork is enabled, this will set to ClusterFirstWithHostNet automatically + dnsPolicy: ClusterFirst + + # Use hostPort + # hostPort: 9090 + + # Use portName + portName: "" + + ## Vertical Pod Autoscaler config + ## Ref: https://github.com/kubernetes/autoscaler/tree/master/vertical-pod-autoscaler + verticalAutoscaler: + ## If true a VPA object will be created for the controller (either StatefulSet or Deployemnt, based on above configs) + enabled: false + # updateMode: "Auto" + # containerPolicies: + # - containerName: 'prometheus-server' + + # Custom DNS configuration to be added to prometheus server pods + dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 + + ## Security context to be added to server pods + ## + securityContext: + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + fsGroup: 65534 + + ## Security context to be added to server container + ## + containerSecurityContext: {} + + service: + ## If false, no Service will be created for the Prometheus server + ## + enabled: true + + annotations: {} + labels: {} + clusterIP: "" + + ## List of IP addresses at which the Prometheus server service is available + ## Ref: https://kubernetes.io/docs/concepts/services-networking/service/#external-ips + ## + externalIPs: [] + + loadBalancerIP: "" + loadBalancerSourceRanges: [] + servicePort: 80 + sessionAffinity: None + type: ClusterIP + + ## Enable gRPC port on service to allow auto discovery with thanos-querier + gRPC: + enabled: false + servicePort: 10901 + # nodePort: 10901 + + ## If using a statefulSet (statefulSet.enabled=true), configure the + ## service to connect to a specific replica to have a consistent view + ## of the data. + statefulsetReplica: + enabled: false + replica: 0 + + ## Additional port to define in the Service + additionalPorts: [] + # additionalPorts: + # - name: authenticated + # port: 8081 + # targetPort: 8081 + + ## Prometheus server pod termination grace period + ## + terminationGracePeriodSeconds: 300 + + ## Prometheus data retention period (default if not specified is 15 days) + ## + retention: "15d" + + ## Prometheus' data retention size. Supported units: B, KB, MB, GB, TB, PB, EB. + ## + retentionSize: "" + +## Prometheus server ConfigMap entries for rule files (allow prometheus labels interpolation) +ruleFiles: {} + +## Prometheus server ConfigMap entries for scrape_config_files +## (allows scrape configs defined in additional files) +## +scrapeConfigFiles: [] + +## Prometheus server ConfigMap entries +## +serverFiles: + ## Alerts configuration + ## Ref: https://prometheus.io/docs/prometheus/latest/configuration/alerting_rules/ + alerting_rules.yml: {} + # groups: + # - name: Instances + # rules: + # - alert: InstanceDown + # expr: up == 0 + # for: 5m + # labels: + # severity: page + # annotations: + # description: '{{ $labels.instance }} of job {{ $labels.job }} has been down for more than 5 minutes.' + # summary: 'Instance {{ $labels.instance }} down' + ## DEPRECATED DEFAULT VALUE, unless explicitly naming your files, please use alerting_rules.yml + alerts: {} + + ## Records configuration + ## Ref: https://prometheus.io/docs/prometheus/latest/configuration/recording_rules/ + recording_rules.yml: {} + ## DEPRECATED DEFAULT VALUE, unless explicitly naming your files, please use recording_rules.yml + rules: {} + + prometheus.yml: + rule_files: + - /etc/config/recording_rules.yml + - /etc/config/alerting_rules.yml + ## Below two files are DEPRECATED will be removed from this default values file + - /etc/config/rules + - /etc/config/alerts + + scrape_configs: + - job_name: prometheus + static_configs: + - targets: + - localhost:9090 + + # A scrape configuration for running Prometheus on a Kubernetes cluster. + # This uses separate scrape configs for cluster components (i.e. API server, node) + # and services to allow each to use different authentication configs. + # + # Kubernetes labels will be added as Prometheus labels on metrics via the + # `labelmap` relabeling action. + + # Scrape config for API servers. + # + # Kubernetes exposes API servers as endpoints to the default/kubernetes + # service so this uses `endpoints` role and uses relabelling to only keep + # the endpoints associated with the default/kubernetes service using the + # default named port `https`. This works for single API server deployments as + # well as HA API server deployments. + - job_name: 'kubernetes-apiservers' + + kubernetes_sd_configs: + - role: endpoints + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + # Keep only the default/kubernetes service endpoints for the https port. This + # will add targets for each API server which Kubernetes adds an endpoint to + # the default/kubernetes service. + relabel_configs: + - source_labels: [__meta_kubernetes_namespace, __meta_kubernetes_service_name, __meta_kubernetes_endpoint_port_name] + action: keep + regex: default;kubernetes;https + + - job_name: 'kubernetes-nodes' + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + kubernetes_sd_configs: + - role: node + + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_node_label_(.+) + - target_label: __address__ + replacement: kubernetes.default.svc:443 + - source_labels: [__meta_kubernetes_node_name] + regex: (.+) + target_label: __metrics_path__ + replacement: /api/v1/nodes/$1/proxy/metrics + + + - job_name: 'kubernetes-nodes-cadvisor' + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + kubernetes_sd_configs: + - role: node + + # This configuration will work only on kubelet 1.7.3+ + # As the scrape endpoints for cAdvisor have changed + # if you are using older version you need to change the replacement to + # replacement: /api/v1/nodes/$1:4194/proxy/metrics + # more info here https://github.com/coreos/prometheus-operator/issues/633 + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_node_label_(.+) + - target_label: __address__ + replacement: kubernetes.default.svc:443 + - source_labels: [__meta_kubernetes_node_name] + regex: (.+) + target_label: __metrics_path__ + replacement: /api/v1/nodes/$1/proxy/metrics/cadvisor + + # Metric relabel configs to apply to samples before ingestion. + # [Metric Relabeling](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#metric_relabel_configs) + # metric_relabel_configs: + # - action: labeldrop + # regex: (kubernetes_io_hostname|failure_domain_beta_kubernetes_io_region|beta_kubernetes_io_os|beta_kubernetes_io_arch|beta_kubernetes_io_instance_type|failure_domain_beta_kubernetes_io_zone) + + # Scrape config for service endpoints. + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/scrape`: Only scrape services that have a value of + # `true`, except if `prometheus.io/scrape-slow` is set to `true` as well. + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: If the metrics are exposed on a different port to the + # service then set this appropriately. + # * `prometheus.io/param_`: If the metrics endpoint uses parameters + # then you can set any parameter + - job_name: 'kubernetes-service-endpoints' + honor_labels: true + + kubernetes_sd_configs: + - role: endpoints + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape] + action: keep + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape_slow] + action: drop + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scheme] + action: replace + target_label: __scheme__ + regex: (https?) + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_service_annotation_prometheus_io_port] + action: replace + target_label: __address__ + regex: (.+?)(?::\d+)?;(\d+) + replacement: $1:$2 + - action: labelmap + regex: __meta_kubernetes_service_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + action: replace + target_label: service + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + + # Scrape config for slow service endpoints; same as above, but with a larger + # timeout and a larger interval + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/scrape-slow`: Only scrape services that have a value of `true` + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: If the metrics are exposed on a different port to the + # service then set this appropriately. + # * `prometheus.io/param_`: If the metrics endpoint uses parameters + # then you can set any parameter + - job_name: 'kubernetes-service-endpoints-slow' + honor_labels: true + + scrape_interval: 5m + scrape_timeout: 30s + + kubernetes_sd_configs: + - role: endpoints + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape_slow] + action: keep + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scheme] + action: replace + target_label: __scheme__ + regex: (https?) + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_service_annotation_prometheus_io_port] + action: replace + target_label: __address__ + regex: (.+?)(?::\d+)?;(\d+) + replacement: $1:$2 + - action: labelmap + regex: __meta_kubernetes_service_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + action: replace + target_label: service + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + + - job_name: 'prometheus-pushgateway' + honor_labels: true + + kubernetes_sd_configs: + - role: service + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_probe] + action: keep + regex: pushgateway + + # Example scrape config for probing services via the Blackbox Exporter. + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/probe`: Only probe services that have a value of `true` + - job_name: 'kubernetes-services' + honor_labels: true + + metrics_path: /probe + params: + module: [http_2xx] + + kubernetes_sd_configs: + - role: service + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_probe] + action: keep + regex: true + - source_labels: [__address__] + target_label: __param_target + - target_label: __address__ + replacement: blackbox + - source_labels: [__param_target] + target_label: instance + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + target_label: service + + # Example scrape config for pods + # + # The relabeling allows the actual pod scrape endpoint to be configured via the + # following annotations: + # + # * `prometheus.io/scrape`: Only scrape pods that have a value of `true`, + # except if `prometheus.io/scrape-slow` is set to `true` as well. + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: Scrape the pod on the indicated port instead of the default of `9102`. + - job_name: 'kubernetes-pods' + honor_labels: true + + kubernetes_sd_configs: + - role: pod + + relabel_configs: + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape] + action: keep + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape_slow] + action: drop + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scheme] + action: replace + regex: (https?) + target_label: __scheme__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_port, __meta_kubernetes_pod_ip] + action: replace + regex: (\d+);(([A-Fa-f0-9]{1,4}::?){1,7}[A-Fa-f0-9]{1,4}) + replacement: '[$2]:$1' + target_label: __address__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_port, __meta_kubernetes_pod_ip] + action: replace + regex: (\d+);((([0-9]+?)(\.|$)){4}) + replacement: $2:$1 + target_label: __address__ + - action: labelmap + regex: __meta_kubernetes_pod_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_pod_name] + action: replace + target_label: pod + - source_labels: [__meta_kubernetes_pod_phase] + regex: Pending|Succeeded|Failed|Completed + action: drop + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + + # Example Scrape config for pods which should be scraped slower. An useful example + # would be stackriver-exporter which queries an API on every scrape of the pod + # + # The relabeling allows the actual pod scrape endpoint to be configured via the + # following annotations: + # + # * `prometheus.io/scrape-slow`: Only scrape pods that have a value of `true` + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: Scrape the pod on the indicated port instead of the default of `9102`. + - job_name: 'kubernetes-pods-slow' + honor_labels: true + + scrape_interval: 5m + scrape_timeout: 30s + + kubernetes_sd_configs: + - role: pod + + relabel_configs: + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape_slow] + action: keep + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scheme] + action: replace + regex: (https?) + target_label: __scheme__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_port, __meta_kubernetes_pod_ip] + action: replace + regex: (\d+);(([A-Fa-f0-9]{1,4}::?){1,7}[A-Fa-f0-9]{1,4}) + replacement: '[$2]:$1' + target_label: __address__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_port, __meta_kubernetes_pod_ip] + action: replace + regex: (\d+);((([0-9]+?)(\.|$)){4}) + replacement: $2:$1 + target_label: __address__ + - action: labelmap + regex: __meta_kubernetes_pod_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_pod_name] + action: replace + target_label: pod + - source_labels: [__meta_kubernetes_pod_phase] + regex: Pending|Succeeded|Failed|Completed + action: drop + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + +# adds additional scrape configs to prometheus.yml +# must be a string so you have to add a | after extraScrapeConfigs: +# example adds prometheus-blackbox-exporter scrape config +extraScrapeConfigs: "" + # - job_name: 'prometheus-blackbox-exporter' + # metrics_path: /probe + # params: + # module: [http_2xx] + # static_configs: + # - targets: + # - https://example.com + # relabel_configs: + # - source_labels: [__address__] + # target_label: __param_target + # - source_labels: [__param_target] + # target_label: instance + # - target_label: __address__ + # replacement: prometheus-blackbox-exporter:9115 + +# Adds option to add alert_relabel_configs to avoid duplicate alerts in alertmanager +# useful in H/A prometheus with different external labels but the same alerts +alertRelabelConfigs: {} + # alert_relabel_configs: + # - source_labels: [dc] + # regex: (.+)\d+ + # target_label: dc + +networkPolicy: + ## Enable creation of NetworkPolicy resources. + ## + ## Customized for K10 + enabled: true + +# Force namespace of namespaced resources +forceNamespace: "" + +# Extra manifests to deploy as an array +extraManifests: [] + # - | + # apiVersion: v1 + # kind: ConfigMap + # metadata: + # labels: + # name: prometheus-extra + # data: + # extra-data: "value" + +# Configuration of subcharts defined in Chart.yaml + +## alertmanager sub-chart configurable values +## Please see https://github.com/prometheus-community/helm-charts/tree/main/charts/alertmanager +## +alertmanager: + ## If false, alertmanager will not be installed + ## + ## Customized for K10 + enabled: false + + persistence: + size: 2Gi + + podSecurityContext: + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + fsGroup: 65534 + +## kube-state-metrics sub-chart configurable values +## Please see https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics +## +kube-state-metrics: + ## If false, kube-state-metrics sub-chart will not be installed + ## + ## Customized for K10 + enabled: false + +## prometheus-node-exporter sub-chart configurable values +## Please see https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-node-exporter +## +prometheus-node-exporter: + ## If false, node-exporter will not be installed + ## + ## Customized for K10 + enabled: false + + rbac: + pspEnabled: false + + containerSecurityContext: + allowPrivilegeEscalation: false + +## prometheus-pushgateway sub-chart configurable values +## Please see https://github.com/prometheus-community/helm-charts/tree/main/charts/prometheus-pushgateway +## +prometheus-pushgateway: + ## If false, pushgateway will not be installed + ## + ## Customized for K10 + enabled: false + + # Optional service annotations + serviceAnnotations: + prometheus.io/probe: pushgateway diff --git a/charts/kasten/k10/7.0.1401/config.json b/charts/kasten/k10/7.0.1401/config.json new file mode 100644 index 000000000..e69de29bb diff --git a/charts/kasten/k10/7.0.1401/eula.txt b/charts/kasten/k10/7.0.1401/eula.txt new file mode 100644 index 000000000..19f9fc076 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/eula.txt @@ -0,0 +1,459 @@ +KASTEN END USER LICENSE AGREEMENT + +This End User License Agreement is a binding agreement between Kasten, Inc., a +Delaware Corporation ("Kasten"), and you ("Licensee"), and establishes the terms +under which Licensee may use the Software and Documentation (as defined below), +including without limitation terms and conditions relating to license grant, +intellectual property rights, disclaimers /exclusions / limitations of warranty, +indemnity and liability, governing law and limitation periods. All components +collectively are referred to herein as the "Agreement." + +LICENSEE ACKNOWLEDGES IT HAS HAD THE OPPORTUNITY TO REVIEW THE AGREEMENT, PRIOR +TO ACCEPTANCE OF THIS AGREEMENT. LICENSEE'S ACCEPTANCE OF THIS AGREEMENT IS +EVIDENCED BY LICENSEE'S DOWNLOADING, COPYING, INSTALLING OR USING THE KASTEN +SOFTWARE. IF YOU ARE ACTING ON BEHALF OF A COMPANY, YOU REPRESENT THAT YOU ARE +AUTHORIZED TO BIND THE COMPANY. IF YOU DO NOT AGREE TO ALL TERMS OF THIS +AGREEMENT, DO NOT DOWNLOAD, COPY, INSTALL, OR USE THE SOFTWARE, AND PERMANENTLY +DELETE THE SOFTWARE. + +1. DEFINITIONS + +1.1 "Authorized Persons" means trained technical employees and contractors of +Licensee who are subject to a written agreement with Licensee that includes use +and confidentiality restrictions that are at least as protective as those set +forth in this Agreement. + +1.2 "Authorized Reseller" means a distributor or reseller, including cloud +computing platform providers, authorized by Kasten to resell licenses to the +Software through the channel through or in the territory in which Licensee is +purchasing. + +1.3 "Confidential Information" means all non-public information disclosed in +written, oral or visual form by either party to the other. Confidential +Information may include, but is not limited to, services, pricing information, +computer programs, source code, names and expertise of employees and +consultants, know-how, and other technical, business, financial and product +development information. "Confidential Information" does not include any +information that the receiving party can demonstrate by its written records (1) +was rightfully known to it without obligation of confidentiality prior to its +disclosure hereunder by the disclosing party; (2) is or becomes publicly known +through no wrongful act of the receiving party; (3) has been rightfully received +without obligation of confidentiality from a third party authorized to make such +a disclosure; or (4) is independently developed by the receiving party without +reference to confidential information disclosed hereunder. + +1.4 "Documentation" means any administration guides, installation and user +guides, and release notes that are provided by Kasten to Licensee with the +Software. + +1.5 "Intellectual Property Rights" means patents, design patents, copyrights, +trademarks, Confidential Information, know-how, trade secrets, moral rights, and +any other intellectual property rights recognized in any country or jurisdiction +in the world. + +1.6 "Node" means a single physical or virtual computing machine recognizable by +the Software as a unique device. Nodes must be owned or leased by Licensee or an +entity controlled by, controlling or under common control with Licensee. + +1.7 "Edition" means a unique identifier for each distinct product that is made +available by Kasten and that can be licensed, including summary information +regarding any associated functionality, features, or restrictions specific to +the Edition. + +1.8 "Open Source Software" means software delivered to Licensee hereunder that +is subject to the provisions of any open source license agreement. + +1.9 "Purchase Agreement" means a separate commercial agreement, if applicable, +between Kasten and the Licensee that contains the terms for the licensing of a +specific Edition of the Software. + +1.10 "Software" means any and all software product Editions licensed to Licensee +under this Agreement, all as developed by Kasten and delivered to Licensee +hereunder. Software also includes any Updates provided by Kasten to Licensee. +For the avoidance of doubt, the definition of Software shall exclude any +Third-Party Software and Open Source Software. + +1.11 "Third-Party Software" means certain software Kasten licenses from third +parties and provides to Licensee with the Software, which may include Open +Source Software. + +1.12 "Update" means a revision of the Software that Kasten makes available to +customers at no additional cost. The Update includes, if and when applicable and +available, bug fix patches, maintenance release, minor release, or new major +releases. Updates are limited only to the Software licensed by Licensee, and +specifically exclude new product offerings, features, options or functionality +of the Software that Kasten may choose to license separately, or for an +additional fee. + +1.13 "Use" means to install activate the processing capabilities of the +Software, load, execute, access, employ the Software, or display information +resulting from such capabilities. + + +2. LICENSE GRANT AND RESTRICTIONS + +2.1 Enterprise License. Subject to Licensee"s compliance with the terms and +conditions of this Agreement (including any additional restrictions on +Licensee"s use of the Software set forth in the Purchase Agreement, if one +exists, between Licensee and Kasten), Kasten grants to Licensee a non-exclusive, +non-transferable (except in connection with a permitted assignment of this +Agreement under Section 14.10 (Assignment), non-sublicensable, limited term +license to install and use the Software, in object code form only, solely for +Licensee"s use, unless terminated in accordance with Section 4 (Term and +Termination). + +2.2 Starter License. This section shall only apply when the Licensee licenses +Starter Edition of the Software. The license granted herein is for a maximum of +5 Nodes and for a period of 12 months from the date of the Software release that +embeds the specific license instance. Updating to a newer Software (minor or +major) release will always extend the validity of the license by 12 months. If +the Licensee wishes to upgrade to an Enterprise License instead, the Licensee +will have to enter into a Purchase Agreement with Kasten which will supersede +this Agreement. The Licensee is required to provide accurate email and company +information, if representing a company, when accepting this Agreement. Under no +circumstances will a Starter License be construed to mean that the Licensee is +authorized to distribute the Software to any third party for any reason +whatsoever. + +2.3 Evaluation License. This section shall only apply when the Licensee has +licensed the Software for an initial evaluation period. The license granted +herein is valid only one time 30 days, starting from date of installation, +unless otherwise explicitly designated by Kasten ("Evaluation Period"). Under +this license the Software can only be used for evaluation purposes. Under no +circumstances will an Evaluation License be construed to mean that the Licensee +is authorized to distribute the Software to any third party for any reason +whatsoever. If the Licensee wishes to upgrade to an Enterprise License instead, +the Licensee will have to enter into a Purchase Agreement with Kasten which will +supersede this Agreement.. If the Licensee does not wish to upgrade to an +Enterprise License at the end of the Evaluation Period the Licensee"s rights +under the Agreement shall terminate, and the Licensee shall delete all Kasten +Software. + +2.4 License Restrictions. Except to the extent permitted under this Agreement, +Licensee will not nor will Licensee allow any third party to: (i) copy, modify, +adapt, translate or otherwise create derivative works of the Software or the +Documentation; (ii) reverse engineer, decompile, disassemble or otherwise +attempt to discover the source code of the Software; (iii) rent, lease, sell, +assign or otherwise transfer rights in or to the Software or Documentation; (iv) +remove any proprietary notices or labels from the Software or Documentation; (v) +publicly disseminate performance information or analysis (including, without +limitation, benchmarks) relating to the Software. Licensee will comply with all +applicable laws and regulations in Licensee"s use of and access to the Software +and Documentation. + +2.5 Responsibility for Use. The Software and Documentation may be used only by +Authorized Persons and in conformance with this Agreement. Licensee shall be +responsible for the proper use and protection of the Software and Documentation +and is responsible for: (i) installing, managing, operating, and physically +controlling the Software and the results obtained from using the Software; (ii) +using the Software within the operating environment specified in the +Documentation; and; (iii) establishing and maintaining such recovery and data +protection and security procedures as necessary for Licensee's service and +operation and/or as may be specified by Kasten from time to time. + +2.6 United States Government Users. The Software licensed under this Agreement +is "commercial computer software" as that term is described in DFAR +252.227-7014(a)(1). If acquired by or on behalf of a civilian agency, the U.S. +Government acquires this commercial computer software and/or commercial computer +software documentation subject to the terms and this Agreement as specified in +48 C.F.R. 12.212 (Computer Software) and 12.211 (Technical Data) of the Federal +Acquisition Regulations ("FAR") and its successors. If acquired by or on behalf +of any agency within the Department of Defense ("DOD"), the U.S. Government +acquires this commercial computer software and/or commercial computer software +documentation subject to the terms of this Agreement as specified in 48 C.F.R. +227.7202 of the DOD FAR Supplement and its successors. + + +3. SUPPORT + +3.1 During the Term (as defined below) and subject to Licensee’s compliance +with the terms and conditions of this Agreement, Licensee may submit queries and +requests for support for Enterprise Licenses by submitting Service Requests via Veeam +Support Portal (https://my.veeam.com). Support is not provided for Starter and Evaluation +Licenses. Licensee shall be entitled to the support service-level agreement specified +in the relevant order form or purchase order (“Order Form”) between Licensee and the +Reseller and as set forth in Kasten’s Support Policy, a copy of which can be found +at https://www.kasten.io/support-services-policy. Licensee shall also be permitted to +download and install all Updates released by Kasten during the Term and made generally +available to users of the Software. Software versions with all updates and upgrades +installed is supported for six months from the date of release of that version. + +3.2 Starter Edition Support. If the Licensee has licensed Starter Edition of +the Software, you will have access to the Kasten K10 Support Community +(https://community.veeam.com/groups/kasten-k10-support-92), but Kasten cannot guarantee +a service level of any sort. Should a higher level of support be needed, Licensee has +the option to consider entering into a Purchase Agreement with Kasten for licensing a +different Edition of the Software. + + + +4. TERM AND TERMINATION + +4.1 Term. The term of this Agreement, except for Starter and Evaluation +Licenses, shall commence on the Effective Date and shall, unless terminated +earlier in accordance with the provisions of Section 4.2 below, remain in force +for the Subscription Period as set forth in the applicable Order Form(s) (the +"Term"). The parties may extend the Term of this Agreement beyond the +Subscription Period by executing additional Order Form(s) and Licensee"s payment +of additional licensing fees. The term of this Agreement for the Starter and +Evaluation Licenses will coincide with the term for Starter Edition (as stated +in section 2.2) and the term for Evaluation Period (as stated in section 2.3), +respectively + +4.2 Termination. Either party may immediately terminate this +Agreement and the licenses granted hereunder if the other party (1) becomes +insolvent and"becomes unwilling or unable to meet its obligations under this +Agreement, (2) files a petition in bankruptcy, (3) is subject to the filing of +an involuntary petition for bankruptcy which is not rescinded within a period of +forty-five (45) days, (4) fails to cure a material breach of any material term +or condition of this Agreement within thirty (30) days of receipt of written +notice specifying such breach, or (5) materially breaches its obligations of +confidentiality hereunder. + +4.3 Effects of Termination. Upon expiration or +termination of this Agreement for any reason, (i) any amounts owed to Kasten +under this Agreement will be immediately due and payable; (ii) all licensed +rights granted in this Agreement will immediately cease; and (iii) Licensee will +promptly discontinue all use of the Software and Documentation and return to +Kasten any Kasten Confidential Information in Licensee"s possession or control. + +4.4 Survival. The following Sections of this Agreement will remain in effect +following the expiration or termination of these General Terms for any reason: +4.3 (Effects of Termination), 4.4 (Survival), 5 (Third Party Software) 5 +(Confidentiality), 9 (Ownership), 10.2 (Third-Party Software), 10.3 (Warranty +Disclaimer), 11 (Limitations of Liability), 12.2 (Exceptions to Kasten +Obligation), 13 (Export) and 14 (General). + + +5. THIRD PARTY AND OPEN SOURCE SOFTWARE Certain Third-Party Software or Open +Source Software (Kasten can provide a list upon request) that may be provided +with the Software may be subject to various other terms and conditions imposed +by the licensors of such Third-Party Software or Open Source Software. The +terms of Licensee"s use of the Third-Party Software or Open Source Software is +subject to and governed by the respective Third-Party Software and Open Source +licenses, except that this Section 5 (Third-Party Software), Section 10.2 (Third +Party Software), 10.3 (Warranty Disclaimer), Section 11 (Limitations of +Liability), and Section 14 (General) of this Agreement also govern Licensee"s +use of the Third-Party Software. To the extent applicable to Licensee"s use of +such Third-Party Software and Open Source, Licensee agrees to comply with the +terms and conditions contained in all such Third-Party Software and Open Source +licenses. + + +6. CONFIDENTIALITY Neither party will use any Confidential Information of the +other party except as expressly permitted by this Agreement or as expressly +authorized in writing by the disclosing party. The receiving party shall use +the same degree of care to protect the disclosing party"s Confidential +Information as it uses to protect its own Confidential Information of like +nature, but in no circumstances less than a commercially reasonable standard of +care. The receiving party may not disclose the disclosing party"s Confidential +Information to any person or entity other than to (i) (a) Authorized Persons in +the case the receiving party is Licensee, and (b) Kasten"s employees and +contractors in the case the receiving party is Kasten, and (ii) who need access +to such Confidential Information solely for the purpose of fulfilling that +party"s obligations or exercising that party"s rights hereunder. The foregoing +obligations will not restrict the receiving party from disclosing Confidential +Information of the disclosing party: (1) pursuant to the order or requirement of +a court, administrative agency, or other governmental body, provided that the +receiving party required to make such a disclosure gives reasonable notice to +the disclosing party prior to such disclosure; and (2) on a confidential basis +to its legal and financial advisors. Kasten may identify Licensee in its +customer lists in online and print marketing materials. + + +7. FEES Fees for Enterprise License shall be set forth in separate Order Form(s) +attached to a Purchase Agreement, between the Licensee and Kasten. + +If Licensee has obtained the Software through an Authorized Reseller, fees for +licensing shall be invoiced directly by the Authorized Reseller. + +If no Purchase Agreement exists, during the term of this Agreement, Kasten +shall license the Starter Edition only and no other Edition of the Software +"at no charge" to Licensee. + + +8. USAGE DATA Kasten may collect, accumulate, and aggregate certain usage +statistics in order to analyze usage of the Software, make improvements, and +potentially develop new products. Kasten may use aggregated anonymized data for +any purpose that Kasten, at its own discretion, may consider appropriate. + + +9. OWNERSHIP As between Kasten and Licensee, all right, title and interest in +the Software, Documentation and any other Kasten materials furnished or made +available hereunder, all modifications and enhancements thereof, and all +suggestions, ideas and feedback proposed by Licensee regarding the Software and +Documentation, including all copyright rights, patent rights and other +Intellectual Property Rights in each of the foregoing, belong to and are +retained solely by Kasten or Kasten"s licensors and providers, as applicable. +Licensee hereby does and will irrevocably assign to Kasten all evaluations, +ideas, feedback and suggestions made by Licensee to Kasten regarding the +Software and Documentation (collectively, "Feedback") and all Intellectual +Property Rights in and to the Feedback. Except as expressly provided herein, no +licenses of any kind are granted hereunder, whether by implication, estoppel, or +otherwise. + + +10. LIMITED WARRANTY AND DISCLAIMERS + +10.1 Limited Warranty. Kasten warrants for a period of thirty (30) days from +the Effective Date that the Software will materially conform to Kasten"s +then-current Documentation (the "Warranty Period") when properly installed on a +computer for which a license is granted hereunder. Licensee"s exclusive remedy +for a breach of this Section 10.1 is that Kasten shall, at its option, use +commercially reasonable efforts to correct or replace the Software, or refund +all or a portion of the fees paid by Licensee pursuant to the Purchase +Agreement. Kasten, in its sole discretion, may revise this limited warranty from +time to time. + +10.2 Third-Party Software. Except as expressly set forth in this Agreement, +Third-Party Software (including any Open Source Software) are provided on an +"as-is" basis at the sole risk of Licensee. Notwithstanding any language to the +contrary in this Agreement, Kasten makes no express or implied warranties of any +kind with respect to Third-Party Software provided to Licensee and shall not be +liable for any damages regarding the use or operation of the Third-Party +Software furnished under this Agreement. Any and all express or implied +warranties, if any, arising from the license of Third-Party Software shall be +those warranties running from the third party manufacturer or licensor to +Licensee. + +10.3 Warranty Disclaimer. EXCEPT FOR THE LIMITED WARRANTY PROVIDED ABOVE, +KASTEN AND ITS SUPPLIERS MAKE NO WARRANTY OF ANY KIND, WHETHER EXPRESS, IMPLIED, +STATUTORY OR OTHERWISE, RELATING TO THE SOFTWARE OR TO KASTEN"S MAINTENANCE, +PROFESSIONAL OR OTHER SERVICES. KASTEN SPECIFICALLY DISCLAIMS ALL IMPLIED +WARRANTIES OF DESIGN, MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE +AND NON-INFRINGEMENT. KASTEN AND ITS SUPPLIERS AND LICENSORS DO NOT WARRANT OR +REPRESENT THAT THE SOFTWARE WILL BE FREE FROM BUGS OR THAT ITS USE WILL BE +UNINTERRUPTED OR ERROR-FREE. THIS DISCLAIMER SHALL APPLY NOTWITHSTANDING THE +FAILURE OF THE ESSENTIAL PURPOSE OF ANY LIMITED REMEDY PROVIDED HEREIN. EXCEPT +AS STATED ABOVE, KASTEN AND ITS SUPPLIERS PROVIDE THE SOFTWARE ON AN "AS IS" +BASIS. KASTEN PROVIDES NO WARRANTIES WITH RESPECT TO THIRD PARTY SOFTWARE AND +OPEN SOURCE SOFTWARE. + + +11. LIMITATIONS OF LIABILITY + +11.1 EXCLUSION OF CERTAIN DAMAGES. EXCEPT FOR BREACHES OF SECTION 6 +(CONFIDENTIALITY) OR SECTION 9 (OWNERSHIP), IN NO EVENT WILL EITHER PARTY BE +LIABLE FOR ANY INDIRECT, CONSEQUENTIAL, EXEMPLARY, SPECIAL, INCIDENTAL OR +RELIANCE DAMAGES, INCLUDING ANY LOST DATA, LOSS OF USE AND LOST PROFITS, ARISING +FROM OR RELATING TO THIS AGREEMENT, THE SOFTWARE OR DOCUMENTATION, EVEN IF SUCH +PARTY KNEW OR SHOULD HAVE KNOWN OF THE POSSIBILITY OF, OR COULD REASONABLY HAVE +PREVENTED, SUCH DAMAGES. + +11.2 LIMITATION OF DAMAGES. EXCEPT FOR THE BREACHES OF SECTION 6 +(CONFIDENTIALITY) OR SECTION 9 (OWNERSHIP), EACH PARTY"S TOTAL CUMULATIVE +LIABILITY ARISING FROM OR RELATED TO THIS AGREEMENT OR THE SOFTWARE, +DOCUMENTATION, OR SERVICES PROVIDED BY KASTEN, WILL NOT EXCEED THE AMOUNT OF +FEES PAID OR PAYABLE BY LICENSEE FOR THE SOFTWARE, DOCUMENTATION OR SERVICES +GIVING RISE TO THE CLAIM IN THE TWELVE (12) MONTHS FOLLOWING THE EFFECTIVE DATE. +LICENSEE AGREES THAT KASTEN"S SUPPLIERS AND LICENSORS WILL HAVE NO LIABILITY OF +ANY KIND UNDER OR AS A RESULT OF THIS AGREEMENT. IN THE CASE OF KASTEN"S +INDEMNIFICATION OBLIGATIONS, KASTEN"S CUMULATIVE LIABILITY UNDER THIS AGREEMENT +SHALL BE LIMITED TO THE SUM OF THE LICENSE FEES PAID OR PAYABLE BY LICENSEE FOR +THE SOFTWARE, DOCUMENTATION OR SERVICES GIVING RISE TO THE CLAIM IN THE TWELVE +(12) MONTHS FOLLOWING THE EFFECTIVE DATE. + +11.3 THIRD PARTY SOFTWARE. NOTWITHSTANDING ANY LANGUAGE TO THE CONTRARY IN THIS +AGREEMENT, KASTEN SHALL NOT BE LIABLE FOR ANY DAMAGES REGARDING THE USE OR +OPERATION OF ANY THIRD-PARTY SOFTWARE FURNISHED UNDER THIS AGREEMENT. + +11.4 LIMITATION OF ACTIONS. IN NO EVENT MAY LICENSEE BRING ANY CAUSE OF ACTION +RELATED TO THIS AGREEMENT MORE THAN ONE (1) YEAR AFTER THE OCCURRENCE OF THE +EVENT GIVING RISE TO THE LIABILITY. + + +12. EXPORT +The Software, Documentation and related technical data may be subject +to U.S. export control laws, including without limitation the U.S. Export +Administration Act and its associated regulations, and may be subject to export +or import regulations in other countries. Licensee shall comply with all such +regulations and agrees to obtain all necessary licenses to export, re-export, or +import the Software, Documentation and related technical data. + + +13. GENERAL + +13.1 No Agency. Kasten and Licensee each acknowledge and agree that the +relationship established by this Agreement is that of independent contractors, +and nothing contained in this Agreement shall be construed to: (1) give either +party the power to direct or control the daytoday activities of the other; (2) +deem the parties to be acting as partners, joint venturers, coowners or +otherwise as participants in a joint undertaking; or (3) permit either party or +any of either party"s officers, directors, employees, agents or representatives +to create or assume any obligation on behalf of or for the account of the other +party for any purpose whatsoever. + +13.2 Compliance with Laws. Each party agrees to comply with all applicable +laws, regulations, and ordinances relating to their performance hereunder. +Without limiting the foregoing, Licensee warrants and covenants that it will +comply with all then current laws and regulations of the United States and other +jurisdictions relating or applicable to Licensee"s use of the Software and +Documentation including, without limitation, those concerning Intellectual +Property Rights, invasion of privacy, defamation, and the import and export of +Software and Documentation. + +13.3 Force Majeure. Except for the duty to pay money, neither party shall be +liable hereunder by reason of any failure or delay in the performance of its +obligations hereunder on account of strikes, riots, fires, flood, storm, +explosions, acts of God, war, governmental action, earthquakes, or any other +cause which is beyond the reasonable control of such party. + +13.4 Governing Law; Venue and Jurisdiction. This Agreement shall be interpreted +according to the laws of the State of California without regard to or +application of choiceoflaw rules or principles. The parties expressly agree +that the United Nations Convention on Contracts for the International Sale of +Goods and the Uniform Computer Information Transactions Act will not apply. Any +legal action or proceeding arising under this Agreement will be brought +exclusively in the federal or state courts located in Santa Clara County, +California and the parties hereby consent to the personal jurisdiction and venue +therein. + +13.5 Injunctive Relief. The parties agree that monetary damages would not be an +adequate remedy for the breach of certain provisions of this Agreement, +including, without limitation, all provisions concerning infringement, +confidentiality and nondisclosure, or limitation on permitted use of the +Software or Documentation. The parties further agree that, in the event of such +breach, injunctive relief would be necessary to prevent irreparable injury. +Accordingly, either party shall have the right to seek injunctive relief or +similar equitable remedies to enforce such party's rights under the pertinent +provisions of this Agreement, without limiting its right to pursue any other +legal remedies available to it. + +13.6 Entire Agreement and Waiver. This Agreement and any exhibits hereto shall +constitute the entire agreement and contains all terms and conditions between +Kasten and Licensee with respect to the subject matter hereof and all prior +agreements, representations, and statement with respect to such subject matter +are superseded hereby. This Agreement may be changed only by written agreement +signed by both Kasten and Licensee. No failure of either party to exercise or +enforce any of its rights under this Agreement shall act as a waiver of +subsequent breaches; and the waiver of any breach shall not act as a waiver of +subsequent breaches. + +13.7 Severability. In the event any provision of this Agreement is held by a +court or other tribunal of competent jurisdiction to be unenforceable, that +provision will be enforced to the maximum extent permissible under applicable +law and the other provisions of this Agreement will remain in full force and +effect. The parties further agree that in the event such provision is an +essential part of this Agreement, they will begin negotiations for a suitable +replacement provision. + +13.8 Counterparts. This Agreement may be executed in any number of +counterparts, each of which, when so executed and delivered (including by +facsimile), shall be deemed an original, and all of which shall constitute one +and the same agreement. + +13.9 Binding Effect. This Agreement shall be binding upon and shall inure to +the benefit of the respective parties hereto, their respective successors and +permitted assigns. + +13.10 Assignment. Neither party may, without the prior written consent of the +other party (which shall not be unreasonably withheld), assign this Agreement, +in whole or in part, either voluntarily or by operation of law, and any attempt +to do so shall be a material default of this Agreement and shall be void. +Notwithstanding the foregoing, Kasten may assign its rights and benefits and +delegate its duties and obligations under this Agreement without the consent of +Licensee in connection with a merger, reorganization or sale of all or +substantially all relevant assets of the assigning party; in each case provided +that such successor assumes the assigning party"s obligations under this +Agreement. + diff --git a/charts/kasten/k10/7.0.1401/files/favicon.png b/charts/kasten/k10/7.0.1401/files/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..fb617ce12c6949ed2dd1bec208c179644bcec0d4 GIT binary patch literal 1802 zcmY*adt8!d8-9^Q;U~4MX*yHXJYWfUAO(S@qG@7bGxIP_4N#l{pHRT^D`su#Je8*P z)jUsR^SG>~RHnI>c|c94qVrIzB^}VzOi6TLef53s^LwBBdhYvruIstqKb|*(x_>Vm zW(orU0PgQcKB$Qp?W+&b%!hTB(=-9ZJ-F8ksFRr~Gz%&{)SnR;2smi4KA;0K1i)H~ zW&mkSV8c2F09#E20B|YjW3^Q0LlsjB{)n~2`_H@#H6mfm;80#@AO(MvorH>^v192d zK@vwx00;uS1}4#YF$h6YB8!U`5Uti3cn#L3(N>6c3hyhTRcIg;;muB_Bd{n}6vm1K zLm&`@WEum1knH<@yJkhSis$h-cr=>N=cD*8D0Xrj+6jllp)t;AXJD;5qOb(C9W+Ak?F|q7pJffAA*673Y?wmX(E=lC6DR@JYxrwGA?Yh`eb0}ft~q6 zS-@*H1iGV44=Z|xFY0HMKAkSj8vYM_RRZ#kFlhRKW%GYVP4jgFbD><&8I zv^XWN3}J45EHCNcSEWmiz#JQ&b_X2DY7W>@rOIn$xI_AF1rucFE!8<*l-JhgP|zUu z+v2>;RYq7)Qz|9=2ic-oiBNT`YySLfGuu*-{BD{G zZ`#7`+d>_8F5q}{&U~MZ3}zw|pk|7l4gHa4YjiMY7q8V@-1>riyq$4w9JQ$$Ns6tq zxd-``i1XUF(Zhxs5>eTjYk5i4=>QP3)TGI-P};Tjlnsrg>Ob|`j%;5zFDWN%Z%y}> zY-2O{QzLFqHcH_=D4@)S!_(U(fBl~u1%3WwFcSoL%=~ti|K!Gg=!?<&BiS1cWo-Ue zp`<5vcckA#0?*<~V+R9zV$)}H3(GPURF7yqh^aJg<5$e=A_Gg=tKXdR^>+X-8i@05 z-W|YKG@-|>d@Rl0EXk!sWgcb=S=69-eXoeAWmDQo`1zqnC%X~N=Q=;%4Jswh+Z*A5 z_QLU|!F?nmyQ=b@PNZ>RYn00lH7DvV?U|jj4cUb+==2H!O$R1Yl8wWsOWOscs1$Bw zfc<=a`~uD1BPLFG({Y3AcZa2KGL47TpB~`fk?V@~uPwJt!HuqMf)rtIAKWE$l-m4I z-8}Vmw4&_^O#!OXE6P{?)(Ar|cJ$gmhpaFSa!;`8Pg<|Un<-(>JS+8#5^g`R6}5@= zzOv3hZ|;|;hDWxNbDT~90D zl9pZ@$4O0bmu6AVD3Q&bJ}tfmgsRiZViC;Sy?NhWUV}}+6_;d(vWq`)XZE-`>^laO zJaBz=d5^@Wk`L}45!R+%ABk{HD(>Tmhm)Y6qZt&F7+WRS-7>Rl^~aR?HcHXO0`D7- z|2?nnC98m0i_>!5t;H_a{jzpHtzcD`KaNE9%mjUU_f3WQ!tf#S@TpG-m{oq%DKX5c z`Ra+*XdEGM1-sL%7~UOCaiTAM7Ylh*66f)}w|=c*El=@R!izK=A4n%&O>e+nGDg0% tn7HN~o~pnSpxO)Prh(w&4OH~a8i>le=(TeuS4aD2@%PzJuJejc{Re;E|Hl9T literal 0 HcmV?d00001 diff --git a/charts/kasten/k10/7.0.1401/files/kasten-logo.svg b/charts/kasten/k10/7.0.1401/files/kasten-logo.svg new file mode 100644 index 000000000..0d0ef14ee --- /dev/null +++ b/charts/kasten/k10/7.0.1401/files/kasten-logo.svg @@ -0,0 +1,24 @@ + + + + + + diff --git a/charts/kasten/k10/7.0.1401/files/styles.css b/charts/kasten/k10/7.0.1401/files/styles.css new file mode 100644 index 000000000..2d9205711 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/files/styles.css @@ -0,0 +1,113 @@ +.theme-body { + background-color: #efefef; + color: #333; + font-family: 'Source Sans Pro', Helvetica, sans-serif; +} + +.theme-navbar { + background-color: #fff; + box-shadow: 0 2px 2px rgba(0, 0, 0, 0.2); + color: #333; + font-size: 13px; + font-weight: 100; + height: 46px; + overflow: hidden; + padding: 0 10px; +} + +.theme-navbar__logo-wrap { + display: inline-block; + height: 100%; + overflow: hidden; + padding: 10px 15px; + width: 300px; +} + +.theme-navbar__logo { + height: 100%; + max-height: 25px; +} + +.theme-heading { + font-size: 20px; + font-weight: 500; + margin-bottom: 10px; + margin-top: 0; +} + +.theme-panel { + background-color: #fff; + box-shadow: 0 5px 15px rgba(0, 0, 0, 0.5); + padding: 30px; +} + +.theme-btn-provider { + background-color: #fff; + color: #333; + min-width: 250px; +} + +.theme-btn-provider:hover { + color: #999; +} + +.theme-btn--primary { + background-color: #333; + border: none; + color: #fff; + min-width: 200px; + padding: 6px 12px; +} + +.theme-btn--primary:hover { + background-color: #666; + color: #fff; +} + +.theme-btn--success { + background-color: #2FC98E; + color: #fff; + width: 250px; +} + +.theme-btn--success:hover { + background-color: #49E3A8; +} + +.theme-form-row { + display: block; + margin: 20px auto; +} + +.theme-form-input { + border-radius: 4px; + border: 1px solid #CCC; + box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075); + color: #666; + display: block; + font-size: 14px; + height: 36px; + line-height: 1.42857143; + margin: auto; + padding: 6px 12px; + width: 250px; +} + +.theme-form-input:focus, +.theme-form-input:active { + border-color: #66AFE9; + outline: none; +} + +.theme-form-label { + font-size: 13px; + font-weight: 600; + margin: 4px auto; + position: relative; + text-align: left; + width: 250px; +} + +.theme-link-back { + margin-top: 4px; +} diff --git a/charts/kasten/k10/7.0.1401/grafana/dashboards/default/default.json b/charts/kasten/k10/7.0.1401/grafana/dashboards/default/default.json new file mode 100644 index 000000000..2c0e12e59 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/grafana/dashboards/default/default.json @@ -0,0 +1,6337 @@ +{ + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": "-- Grafana --", + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "graphTooltip": 0, + "id": 12, + "links": [], + "liveNow": false, + "panels": [ + { + "collapsed": false, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 53, + "panels": [], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "K10 System Resource Usage", + "type": "row" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 1 + }, + "id": 55, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(rate(process_cpu_seconds_total[5m]))", + "legendFormat": "Total CPU seconds", + "range": true, + "refId": "A" + } + ], + "title": "K10 CPU total seconds ", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "decbytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 1 + }, + "id": 57, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(process_resident_memory_bytes)", + "hide": false, + "legendFormat": "Total memory consumption", + "range": true, + "refId": "C" + } + ], + "title": "K10 total memory consumption", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 9 + }, + "id": 81, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "rate(process_cpu_seconds_total{job=\"httpServiceDiscovery\"}[5m])", + "legendFormat": "{{service}}", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(rate(process_cpu_seconds_total{job=\"k10-pods\"}[5m]))", + "hide": false, + "legendFormat": "executor", + "range": true, + "refId": "B" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(rate(process_cpu_seconds_total{job=\"pushAggregator\"}[5m]))", + "hide": false, + "legendFormat": "ephemeral pods", + "range": true, + "refId": "C" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(rate(process_cpu_seconds_total{job=\"prometheus\"}[5m]))", + "hide": false, + "legendFormat": "prometheus", + "range": true, + "refId": "D" + } + ], + "title": "CPU total seconds per service", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "decbytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 9 + }, + "id": 82, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "process_resident_memory_bytes{job=\"pushAggregator\"}", + "hide": false, + "legendFormat": "ephemeral pods", + "range": true, + "refId": "C" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "process_resident_memory_bytes{job=\"httpServiceDiscovery\"}", + "hide": false, + "legendFormat": "{{service}}", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(process_resident_memory_bytes{job=\"k10-pods\"})", + "hide": false, + "legendFormat": "executor", + "range": true, + "refId": "B" + }, + { + "datasource": "Prometheus", + "editorMode": "builder", + "expr": "sum(process_resident_memory_bytes{job=\"prometheus\"})", + "hide": false, + "legendFormat": "prometheus", + "range": true, + "refId": "D" + } + ], + "title": "Memory consumption by service", + "type": "timeseries" + }, + { + "collapsed": false, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 17 + }, + "id": 18, + "panels": [], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "Applications", + "type": "row" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "yellow", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 5, + "x": 0, + "y": 18 + }, + "id": 24, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Backups Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 5, + "y": 18 + }, + "id": 33, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Backups Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 8, + "y": 18 + }, + "id": 34, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Backups Skipped", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 5, + "x": 13, + "y": 18 + }, + "id": 35, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Restores Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 18, + "y": 18 + }, + "id": 36, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Restores Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 21, + "y": 18 + }, + "id": 23, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Restores Skipped", + "type": "stat" + }, + { + "collapsed": false, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 25 + }, + "id": 16, + "panels": [], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "Cluster", + "type": "row" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "yellow", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 5, + "x": 0, + "y": 26 + }, + "id": 10, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_cluster_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Backups Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 5, + "y": 26 + }, + "id": 19, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_cluster_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Backups Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 8, + "y": 26 + }, + "id": 28, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_backup_cluster_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Backups Skipped", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 5, + "x": 13, + "y": 26 + }, + "id": 21, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_cluster_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Restores Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 18, + "y": 26 + }, + "id": 22, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_cluster_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Restores Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 3, + "x": 21, + "y": 26 + }, + "id": 25, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_restore_cluster_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Cluster Restores Skipped", + "type": "stat" + }, + { + "collapsed": false, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 33 + }, + "id": 31, + "panels": [], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "Backup Exports", + "type": "row" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 5, + "x": 0, + "y": 34 + }, + "id": 38, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_export_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Exports Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 5, + "y": 34 + }, + "id": 29, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_export_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Exports Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 8, + "y": 34 + }, + "id": 20, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_export_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Exports Skipped", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 5, + "x": 13, + "y": 34 + }, + "id": 27, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_import_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Imports Completed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 18, + "y": 34 + }, + "id": 39, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_import_ended_overall{cluster=\"$cluster\", state=~\"failed|cancelled\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Imports Failed", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "#EAB839", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 21, + "y": 34 + }, + "id": 37, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_import_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Imports Skipped", + "type": "stat" + }, + { + "collapsed": false, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 40 + }, + "id": 14, + "panels": [], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "System", + "type": "row" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + }, + "unit": "runs" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 0, + "y": 41 + }, + "id": 12, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_run_ended_overall{cluster=\"$cluster\", state=\"succeeded\"}[$__range])))", + "format": "time_series", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Policy Runs", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "index": 0, + "text": "-" + } + }, + "type": "value" + } + ], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "yellow", + "value": 1 + } + ] + }, + "unit": "runs" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 3, + "y": 41 + }, + "id": 40, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "sum(round(increase(action_run_skipped_overall{cluster=\"$cluster\"}[$__range])))", + "format": "time_series", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Policy Runs Skipped", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 6, + "y": 41 + }, + "id": 6, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "catalog_persistent_volume_disk_space_used_bytes{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Catalog Volume Used", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "max": 100, + "min": 0, + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "yellow", + "value": 70 + }, + { + "color": "orange", + "value": 80 + }, + { + "color": "red", + "value": 90 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 9, + "y": 41 + }, + "id": 2, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": true, + "text": {} + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "100-catalog_persistent_volume_free_space_percent{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Catalog Volume Used Space", + "type": "gauge" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 12, + "y": 41 + }, + "id": 8, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "jobs_persistent_volume_disk_space_used_bytes{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Jobs Volume Used", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "max": 100, + "min": 0, + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "yellow", + "value": 70 + }, + { + "color": "orange", + "value": 80 + }, + { + "color": "red", + "value": 90 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 15, + "y": 41 + }, + "id": 4, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": true, + "text": {} + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "100-jobs_persistent_volume_free_space_percent{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Jobs Volume Used Space", + "type": "gauge" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 18, + "y": 41 + }, + "id": 7, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "logging_persistent_volume_disk_space_used_bytes{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Logging Volume Used", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "max": 100, + "min": 0, + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "yellow", + "value": 70 + }, + { + "color": "orange", + "value": 80 + }, + { + "color": "red", + "value": 90 + } + ] + }, + "unit": "percent" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 21, + "y": 41 + }, + "id": 3, + "options": { + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showThresholdLabels": false, + "showThresholdMarkers": true, + "text": {} + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "100-logging_persistent_volume_free_space_percent{cluster=\"$cluster\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Logging Volume Used Space", + "type": "gauge" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + }, + { + "color": "green", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 0, + "y": 47 + }, + "id": 41, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "compliance_count{state=\"Compliant\"}", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Compliant Applications", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 3, + "y": 47 + }, + "id": 42, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "compliance_count{state=\"NotCompliant\"}", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Non-Compliant Applications", + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "0", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 1 + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 6, + "y": 47 + }, + "id": 43, + "interval": "1m", + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": false, + "expr": "compliance_count{state=\"Unmanaged\"}", + "hide": false, + "interval": "", + "legendFormat": "", + "refId": "B" + } + ], + "title": "Unmanaged Applications", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 12, + "y": 47 + }, + "id": 44, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "snapshot_storage_size_bytes{cluster=\"$cluster\", type=\"physical\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Snapshot Size (Physical)", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 15, + "y": 47 + }, + "id": 45, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "snapshot_storage_size_bytes{cluster=\"$cluster\", type=\"logical\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Snapshot Size (Logical)", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 18, + "y": 47 + }, + "id": 46, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "export_storage_size_bytes{cluster=\"$cluster\", type=\"physical\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Export Size (Physical)", + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "noValue": "-", + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "#ccccdc", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 6, + "w": 3, + "x": 21, + "y": 47 + }, + "id": 47, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "text": {}, + "textMode": "auto" + }, + "pluginVersion": "9.1.5", + "targets": [ + { + "datasource": "Prometheus", + "exemplar": true, + "expr": "export_storage_size_bytes{cluster=\"$cluster\", type=\"logical\"}", + "interval": "", + "legendFormat": "", + "queryType": "randomWalk", + "refId": "A" + } + ], + "title": "Export Size (Logical)", + "type": "stat" + }, + { + "collapsed": true, + "datasource": "Prometheus", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 53 + }, + "id": 49, + "panels": [ + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "red", + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Worker Count" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "dark-red", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 54 + }, + "id": 57, + "interval": "5s", + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(exec_executor_worker_count)", + "legendFormat": "Worker Count", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(exec_active_job_count) OR on() vector(0)", + "hide": false, + "legendFormat": "Worker Load", + "range": true, + "refId": "B" + } + ], + "title": "Executor Worker Load", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineStyle": { + "fill": "solid" + }, + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "s" + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 12, + "y": 54 + }, + "id": 68, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_backup_duration_seconds_sum_overall[5m])) / sum(rate(action_backup_ended_overall[5m]))", + "hide": false, + "legendFormat": "Backup", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_backup_cluster_duration_seconds_overall_sum[5m])) / sum(rate(action_backup_cluster_ended_overall[5m]))", + "hide": false, + "legendFormat": "Backup Cluster", + "range": true, + "refId": "B" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_export_duration_seconds_sum_overall[5m])) / sum(rate(action_export_ended_overall[5m]))", + "hide": false, + "legendFormat": "Export", + "range": true, + "refId": "C" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_import_duration_seconds_sum_overall[5m])) / sum(rate(action_import_ended_overall[5m]))", + "hide": false, + "legendFormat": "Import", + "range": true, + "refId": "D" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_report_duration_seconds_sum_overall[5m])) / sum(rate(action_report_ended_overall[5m]))", + "hide": false, + "legendFormat": "Report", + "range": true, + "refId": "E" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_retire_duration_seconds_sum_overall[5m])) / sum(rate(action_retire_ended_overall[5m]))", + "hide": false, + "legendFormat": "Retire", + "range": true, + "refId": "F" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_restore_duration_seconds_sum_overall[5m])) / sum(rate(action_restore_ended_overall[5m]))", + "hide": false, + "legendFormat": "Restore", + "range": true, + "refId": "G" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(action_restore_cluster_duration_seconds_sum_overall[5m])) / sum(rate(action_restore_cluster_ended_overall[5m]))", + "hide": false, + "legendFormat": "Restore Cluster", + "range": true, + "refId": "H" + } + ], + "title": "Average Action Duration", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 0, + "y": 61 + }, + "id": 74, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_backup_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Backups", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 6, + "y": 61 + }, + "id": 69, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_backup_cluster_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Cluster Backups", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 12, + "y": 61 + }, + "id": 75, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_export_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Exports", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 18, + "y": 61 + }, + "id": 76, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_import_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Imports", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 0, + "y": 68 + }, + "id": 77, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_report_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Reports", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 6, + "y": 68 + }, + "id": 79, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_retire_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Retires", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 12, + "y": 68 + }, + "id": 80, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_restore_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Restores", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "axisSoftMax": 0, + "axisSoftMin": 0, + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "none" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "succeeded" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "failed" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cancelled" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-orange", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "skipped" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "semi-dark-blue", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 18, + "y": 68 + }, + "id": 78, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(round(increase(action_restore_cluster_ended_overall[1m:10s]))) by (state)", + "hide": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Finished Cluster Restores", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "s" + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 24, + "x": 0, + "y": 75 + }, + "id": 63, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(rate(limiter_request_seconds_sum{stage=\"hold\"}[5m])) by (operation) / sum(rate(limiter_request_seconds_count{stage=\"hold\"}[5m])) by (operation) ", + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Rate Limiter - avg operation duration", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "red", + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Limit" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "dark-red", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "inflight" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "green", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "pending" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "yellow", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 7, + "w": 4.8, + "x": 0, + "y": 82 + }, + "id": 51, + "maxPerRow": 6, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "repeat": "operation", + "repeatDirection": "h", + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "limiter_inflight_count{operation=\"$operation\"}", + "legendFormat": "Inflight", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "limiter_pending_count{operation=\"$operation\"}", + "hide": false, + "legendFormat": "Pending", + "range": true, + "refId": "B" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "limiter_inflight_limit_value{operation=\"$operation\"}", + "hide": false, + "legendFormat": "Limit", + "range": true, + "refId": "C" + } + ], + "title": "Rate Limiter - $operation", + "type": "timeseries" + } + ], + "targets": [ + { + "datasource": "Prometheus", + "refId": "A" + } + ], + "title": "Execution Control", + "type": "row" + }, + { + "collapsed": true, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 54 + }, + "id": 84, + "panels": [ + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "percentunit" + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 55 + }, + "id": 86, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(increase(action_export_transferred_bytes[5m:30s]))/sum((increase(action_export_processed_bytes[5m:30s])>0))", + "legendFormat": "Transferred/Processed across all actions", + "range": true, + "refId": "A" + } + ], + "title": "Transferred/Processed Ratio", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "percentunit" + }, + "overrides": [] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 55 + }, + "id": 88, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "(increase(action_export_transferred_bytes[5m:30s])/(increase(action_export_processed_bytes[5m:30s])>0))", + "legendFormat": "{{policy}}:{{app}}", + "range": true, + "refId": "A" + } + ], + "title": "Transferred/Processed Ratio per policy:app", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [ ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "bytes" + }, + "overrides": [ ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 63 + }, + "id": 89, + "options": { + "legend": { + "calcs": [ ], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "increase(action_export_transferred_bytes[5m:30s]) > 0", + "legendFormat": "{{policy}}:{{app}}", + "range": true, + "refId": "A" + } + ], + "title": "Transferred bytes per policy:app", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [ ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "bytes" + }, + "overrides": [ ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 63 + }, + "id": 90, + "options": { + "legend": { + "calcs": [ ], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "increase(action_export_processed_bytes[5m:30s]) > 0", + "legendFormat": "{{policy}}:{{app}}", + "range": true, + "refId": "A" + } + ], + "title": "Processed bytes per policy:app", + "type": "timeseries" + } + ], + "title": "Data reduction", + "type": "row" + }, + { + "collapsed": true, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 55 + }, + "id": 1013, + "panels": [ + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "points", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "stepAfter", + "lineWidth": 1, + "pointSize": 4, + "scaleDistribution": { + "log": 2, + "type": "log" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "s", + "unitScale": true + }, + "overrides": [ + { + "matcher": { + "id": "byRegexp", + "options": "/#.*/" + }, + "properties": [ + { + "id": "unit", + "value": "none" + }, + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "custom.scaleDistribution", + "value": { + "type": "linear" + } + }, + { + "id": "custom.drawStyle", + "value": "line" + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + }, + { + "id": "custom.showPoints", + "value": "never" + }, + { + "id": "custom.axisSoftMin", + "value": 0 + }, + { + "id": "custom.axisLabel", + "value": "# volumes" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#Volumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# Volumes Under Transfer" + }, + { + "id": "custom.lineStyle", + "value": { + "fill": "solid" + } + }, + { + "id": "custom.lineWidth", + "value": 0.4 + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#UploadSessionVolumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# VBR Session Volumes" + }, + { + "id": "custom.lineWidth", + "value": 0 + }, + { + "id": "custom.fillOpacity", + "value": 25 + }, + { + "id": "color", + "value": { + "fixedColor": "dark-blue", + "mode": "shades" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 8 + }, + "id": 1006, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum (max_over_time(data_operation_volume_count{}[2m]))", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#Volumes", + "range": true, + "refId": "VOLUME_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum by (repo_type) (max_over_time(data_upload_session_volume_count{repo_type=\"VBR\"}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#UploadSessionVolumes", + "range": true, + "refId": "VBR_SESSION_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum by (data_format,operation,storage_class,repo_name) (rate(data_operation_normalized_duration_sum{}[2m])) / sum by (data_format,operation,storage_class,repo_name) (rate(data_operation_normalized_duration_count{}[2m]))", + "hide": false, + "instant": false, + "legendFormat": "{{operation}} {{storage_class}}/{{repo_name}} ({{data_format}})", + "range": true, + "refId": "NORMALIZED_DURATION_BY_STORAGE_CLASS_LOC" + } + ], + "title": "Normalized operation duration by storage class, location and data format (time/MiB)", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "points", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "stepAfter", + "lineWidth": 1, + "pointSize": 4, + "scaleDistribution": { + "log": 2, + "type": "log" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "min": 0, + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "s", + "unitScale": true + }, + "overrides": [ + { + "matcher": { + "id": "byRegexp", + "options": "/#.*/" + }, + "properties": [ + { + "id": "unit", + "value": "none" + }, + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "custom.scaleDistribution", + "value": { + "type": "linear" + } + }, + { + "id": "custom.drawStyle", + "value": "line" + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + }, + { + "id": "custom.showPoints", + "value": "never" + }, + { + "id": "custom.axisSoftMin", + "value": 0 + }, + { + "id": "custom.axisLabel", + "value": "# volumes" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#Volumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# Volumes Under Transfer" + }, + { + "id": "custom.lineStyle", + "value": { + "fill": "solid" + } + }, + { + "id": "custom.lineWidth", + "value": 0.4 + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#UploadSessionVolumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# VBR Session Volumes" + }, + { + "id": "custom.lineWidth", + "value": 0 + }, + { + "id": "custom.fillOpacity", + "value": 25 + }, + { + "id": "color", + "value": { + "fixedColor": "dark-blue", + "mode": "shades" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 8 + }, + "id": 1012, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum (max_over_time(data_operation_volume_count{}[2m]))", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#Volumes", + "range": true, + "refId": "VOLUME_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum by (repo_type) (max_over_time(data_upload_session_volume_count{repo_type=\"VBR\"}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#UploadSessionVolumes", + "range": true, + "refId": "VBR_SESSION_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum by (data_format,operation,namespace,pvc_name) (rate(data_operation_duration_sum{}[2m])) / sum by (data_format,operation,namespace,pvc_name) (rate(data_operation_duration_count{}[2m]))", + "hide": false, + "instant": false, + "legendFormat": "{{operation}} {{namespace}}/{{pvc_name}} ({{data_format}})", + "range": true, + "refId": "DURATION_BY_PVC" + } + ], + "title": "Operation duration by pvc and data format", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "points", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "stepAfter", + "lineWidth": 1, + "pointSize": 4, + "scaleDistribution": { + "log": 2, + "type": "log" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "binBps", + "unitScale": true + }, + "overrides": [ + { + "matcher": { + "id": "byRegexp", + "options": "/#.*/" + }, + "properties": [ + { + "id": "unit", + "value": "none" + }, + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "custom.scaleDistribution", + "value": { + "type": "linear" + } + }, + { + "id": "custom.drawStyle", + "value": "line" + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + }, + { + "id": "custom.showPoints", + "value": "never" + }, + { + "id": "custom.axisSoftMin", + "value": 0 + }, + { + "id": "custom.axisLabel", + "value": "# volumes" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#Volumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# Volumes Under Transfer" + }, + { + "id": "custom.lineStyle", + "value": { + "fill": "solid" + } + }, + { + "id": "custom.lineWidth", + "value": 0.4 + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#UploadSessionVolumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# VBR Session Volumes" + }, + { + "id": "custom.lineWidth", + "value": 0 + }, + { + "id": "custom.fillOpacity", + "value": 25 + }, + { + "id": "color", + "value": { + "fixedColor": "dark-blue", + "mode": "shades" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 0, + "y": 16 + }, + "id": 1011, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum (max_over_time(data_operation_volume_count{}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#Volumes", + "range": true, + "refId": "VOLUME_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum by (repo_type) (max_over_time(data_upload_session_volume_count{repo_type=\"VBR\"}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#UploadSessionVolumes", + "range": true, + "refId": "VBR_SESSION_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "avg by (data_format, operation, storage_class, repo_name) (rate(data_operation_bytes{}[$__rate_interval]))", + "hide": false, + "instant": false, + "legendFormat": "{{operation}} {{storage_class}}/{{repo_name}} ({{data_format}})", + "range": true, + "refId": "RATE_BY_STORAGE_CLASS" + } + ], + "title": "Operation transfer rate by storage class, location and data format", + "type": "timeseries" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "points", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "stepAfter", + "lineWidth": 1, + "pointSize": 4, + "scaleDistribution": { + "log": 2, + "type": "log" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "binBps", + "unitScale": true + }, + "overrides": [ + { + "matcher": { + "id": "byRegexp", + "options": "/#.*/" + }, + "properties": [ + { + "id": "unit", + "value": "none" + }, + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "decimals", + "value": 0 + }, + { + "id": "custom.scaleDistribution", + "value": { + "type": "linear" + } + }, + { + "id": "custom.drawStyle", + "value": "line" + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + }, + { + "id": "custom.showPoints", + "value": "never" + }, + { + "id": "custom.axisSoftMin", + "value": 0 + }, + { + "id": "custom.axisLabel", + "value": "# volumes" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#Volumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# Volumes Under Transfer" + }, + { + "id": "custom.lineStyle", + "value": { + "fill": "solid" + } + }, + { + "id": "custom.lineWidth", + "value": 0.4 + }, + { + "id": "custom.lineInterpolation", + "value": "stepAfter" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "#UploadSessionVolumes" + }, + "properties": [ + { + "id": "displayName", + "value": "# VBR Session Volumes" + }, + { + "id": "custom.lineWidth", + "value": 0 + }, + { + "id": "custom.fillOpacity", + "value": 25 + }, + { + "id": "color", + "value": { + "fixedColor": "dark-blue", + "mode": "shades" + } + } + ] + } + ] + }, + "gridPos": { + "h": 8, + "w": 12, + "x": 12, + "y": 16 + }, + "id": 1004, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum (max_over_time(data_operation_volume_count{}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#Volumes", + "range": true, + "refId": "VOLUME_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "expr": "sum by (repo_type) (max_over_time(data_upload_session_volume_count{repo_type=\"VBR\"}[2m]))", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": false, + "legendFormat": "#UploadSessionVolumes", + "range": true, + "refId": "VBR_SESSION_COUNT", + "useBackend": false + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "avg by (data_format, operation, namespace, pvc_name) (rate(data_operation_bytes{}[$__rate_interval]))", + "hide": false, + "instant": false, + "legendFormat": "{{operation}} {{namespace}}/{{pvc_name}} ({{data_format}})", + "range": true, + "refId": "RATE_BY_PVC" + } + ], + "title": "Operation transfer rate by pvc and data format", + "type": "timeseries" + } + ], + "title": "Data transfer operations", + "type": "row" + }, + { + "collapsed": true, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 56 + }, + "id": 1016, + "panels": [ + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 4, + "x": 0, + "y": 57 + }, + "id": 1031, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "repository_data_pruned_bytes > 0", + "format": "time_series", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": false, + "interval": "", + "legendFormat": "__auto", + "range": true, + "refId": "A", + "useBackend": false + } + ], + "title": "Recently recovered storage", + "transformations": [ + { + "id": "reduce", + "options": { + "includeTimeField": false, + "mode": "reduceFields", + "reducers": [ + "lastNotNull" + ] + } + }, + { + "id": "merge", + "options": {} + }, + { + "id": "reduce", + "options": { + "includeTimeField": false, + "mode": "reduceFields", + "reducers": [ + "sum" + ] + } + } + ], + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 8, + "x": 4, + "y": 57 + }, + "id": 1025, + "options": { + "displayMode": "basic", + "maxVizHeight": 300, + "minVizHeight": 16, + "minVizWidth": 8, + "namePlacement": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showUnfilled": true, + "sizing": "auto", + "valueMode": "text" + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "sum by(namespace) (repository_data_pruned_bytes{namespace!=\"kasten-io\"} > 0)", + "format": "time_series", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": false, + "interval": "", + "legendFormat": "__auto", + "range": true, + "refId": "A", + "useBackend": false + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(repository_data_pruned_bytes{namespace=\"kasten-io\"} > 0)", + "hide": false, + "instant": false, + "legendFormat": "metadata", + "range": true, + "refId": "B" + } + ], + "title": "Recently recovered storage per app", + "type": "bargauge" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 4, + "x": 12, + "y": 57 + }, + "id": 1015, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(repository_data_pruned_bytes_total)", + "instant": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Total recovered storage", + "transformations": [ + { + "id": "reduce", + "options": { + "includeTimeField": false, + "mode": "reduceFields", + "reducers": [ + "lastNotNull" + ] + } + } + ], + "type": "stat" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 8, + "x": 16, + "y": 57 + }, + "id": 1018, + "options": { + "displayMode": "basic", + "maxVizHeight": 300, + "minVizHeight": 16, + "minVizWidth": 8, + "namePlacement": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showUnfilled": true, + "sizing": "auto", + "valueMode": "text" + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "exemplar": false, + "expr": "repository_data_pruned_bytes_total{namespace!=\"kasten-io\"} > 0", + "format": "time_series", + "instant": false, + "interval": "", + "legendFormat": "{{namespace}}", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(repository_data_pruned_bytes_total{namespace=\"kasten-io\"} > 0)", + "hide": false, + "instant": false, + "legendFormat": "metadata", + "range": true, + "refId": "B" + } + ], + "title": "Total recovered storage per app", + "type": "bargauge" + }, + { + "datasource": "Prometheus", + "description": "", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 4, + "x": 0, + "y": 62 + }, + "id": 1017, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "percentChangeColorMode": "standard", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(repository_data_scheduled_for_pruning_bytes)", + "instant": false, + "legendFormat": "__auto", + "range": true, + "refId": "A" + } + ], + "title": "Expired data marked for deletion", + "transformations": [ + { + "id": "reduce", + "options": { + "includeTimeField": false, + "mode": "reduceFields", + "reducers": [ + "lastNotNull" + ] + } + } + ], + "type": "stat" + }, + { + "datasource": "Prometheus", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "bytes" + }, + "overrides": [] + }, + "gridPos": { + "h": 5, + "w": 8, + "x": 4, + "y": 62 + }, + "id": 1019, + "options": { + "displayMode": "basic", + "maxVizHeight": 300, + "minVizHeight": 16, + "minVizWidth": 8, + "namePlacement": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showUnfilled": true, + "sizing": "auto", + "valueMode": "text" + }, + "pluginVersion": "11.1.5", + "targets": [ + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "repository_data_scheduled_for_pruning_bytes{namespace!=\"kasten-io\"} > 0", + "instant": false, + "legendFormat": "{{namespace}}", + "range": true, + "refId": "A" + }, + { + "datasource": "Prometheus", + "editorMode": "code", + "expr": "sum(repository_data_scheduled_for_pruning_bytes{namespace=\"kasten-io\"} > 0)", + "hide": false, + "instant": false, + "legendFormat": "metadata", + "range": true, + "refId": "B" + } + ], + "title": "Expired data marked for deletion per app", + "type": "bargauge" + } + ], + "title": "Backup Export Maintenance", + "type": "row" + } + ], + "schemaVersion": 39, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "hide": 2, + "label": "Cluster", + "name": "cluster", + "query": "", + "skipUrlSync": false, + "type": "constant" + }, + { + "current": { + "selected": false, + "text": "All", + "value": "$__all" + }, + "datasource": "Prometheus", + "definition": "limiter_pending_count", + "description": "", + "hide": 2, + "includeAll": true, + "label": "operation", + "multi": false, + "name": "operation", + "options": [], + "query": { + "query": "limiter_pending_count", + "refId": "StandardVariableQuery" + }, + "refresh": 1, + "regex": "/operation=\\\"([\\w]+)\\\"/", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-24h", + "to": "now" + }, + "timepicker": {}, + "timezone": "", + "title": "K10 Dashboard", + "uid": "8Ebb3xS7k", + "version": 3 +} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/license b/charts/kasten/k10/7.0.1401/license new file mode 100644 index 000000000..fb23dbb82 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/license @@ -0,0 +1 @@ +Y3VzdG9tZXJOYW1lOiBzdGFydGVyLWxpY2Vuc2UKZGF0ZUVuZDogJzIxMDAtMDEtMDFUMDA6MDA6MDAuMDAwWicKZGF0ZVN0YXJ0OiAnMjAyMC0wMS0wMVQwMDowMDowMC4wMDBaJwpmZWF0dXJlczogbnVsbAppZDogc3RhcnRlci00ZjE4NDJjMC0wNzQ1LTQxYTUtYWFhNy1hMDFkNzQ4YjFjMzAKcHJvZHVjdDogSzEwCnJlc3RyaWN0aW9uczoKICBub2RlczogJzEwJwpzZXJ2aWNlQWNjb3VudEtleTogbnVsbAp2ZXJzaW9uOiB2MS4wLjAKc2lnbmF0dXJlOiBqT1N5NDNQZG5ZMFVCZitValhOdU1oUEFSb1J2ZkpzWElQWnhBWFNCaGpKbUwxNlNodi8vVzgyV2NMeGZJM25NZTA0TThtRU03eThPcnArQks1ekxpeFd3clpncmZSbTBEaWlELyttRjR5U3l1Rko0QW1neHV6NDhQTmdnU1VyWUM3S1FVcFYxSEJZV1ZaNm9udEJDeE1rVWtkaDVqdzZJdWMzN3lDaktIYy92bWZaenBzTVhybmxUdGhha2RjVVk0azNyVHJDa3VDcnFUMkpjM1o1amFGalZSZW1Zd1NBVXpkRldNazdQdkp3eHVFdE5rNitPV0pCVERQbnNYdldKdjdNc3NneDBJTmdtNUlJWDRVeEVhQWI4QXpTNkMyQ21XQzlhWURFTDg1aEFpeWhONXUwU0tQczA3ZXB0R1VHYmc3cWtPUVN0d0NhcDFKUURvbDVDT0E9PQo= diff --git a/charts/kasten/k10/7.0.1401/questions.yaml b/charts/kasten/k10/7.0.1401/questions.yaml new file mode 100644 index 000000000..713fcb116 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/questions.yaml @@ -0,0 +1,295 @@ +questions: +# ======================== +# SECRETS And Configuration +# ======================== + +### AWS Configuration + +- variable: secrets.awsAccessKeyId + description: "AWS access key ID (required for AWS deployment)" + type: password + label: AWS Access Key ID + required: false + group: "AWS Configuration" + +- variable: secrets.awsSecretAccessKey + description: "AWS access key secret (required for AWS deployment)" + type: password + label: AWS Secret Access Key + required: false + group: "AWS Configuration" + +- variable: secrets.awsIamRole + description: "ARN of the AWS IAM role assumed by K10 to perform any AWS operation." + type: string + label: ARN of the AWS IAM role + required: false + group: "AWS Configuration" + +- variable: awsConfig.assumeRoleDuration + description: "Duration of a session token generated by AWS for an IAM role" + type: string + label: Role Duration + required: false + default: "" + group: "AWS Configuration" + +- variable: awsConfig.efsBackupVaultName + description: "Specifies the AWS EFS backup vault name" + type: string + label: EFS Backup Vault Name + required: false + default: "k10vault" + group: "AWS Configuration" + +### Google Cloud Configuration + +- variable: secrets.googleApiKey + description: "Required If cluster is deployed on Google Cloud" + type: multiline + label: Non-default base64 encoded GCP Service Account key file + required: false + group: "GoogleApi Configuration" + +### Azure Configuration + +- variable: secrets.azureTenantId + description: "Azure tenant ID (required for Azure deployment)" + type: string + label: Tenant ID + required: false + group: "Azure Configuration" + +- variable: secrets.azureClientId + description: "Azure Service App ID" + type: password + label: Service App ID + required: false + group: "Azure Configuration" + +- variable: secrets.azureClientSecret + description: "Azure Service App secret" + type: password + label: Service App secret + required: false + group: "Azure Configuration" + +- variable: secrets.azureResourceGroup + description: "Resource Group name that was created for the Kubernetes cluster" + type: string + label: Resource Group + required: false + group: "Azure Configuration" + +- variable: secrets.azureSubscriptionID + description: "Subscription ID in your Azure tenant" + type: string + label: Subscription ID + required: false + group: "Azure Configuration" + +- variable: secrets.azureResourceMgrEndpoint + description: "Resource management endpoint for the Azure Stack instance" + type: string + label: Resource management endpoint + required: false + group: "Azure Configuration" + +- variable: secrets.azureADEndpoint + description: "Azure Active Directory login endpoint" + type: string + label: Active Directory login endpoint + required: false + group: "Azure Configuration" + +- variable: secrets.azureADResourceID + description: "Azure Active Directory resource ID to obtain AD tokens" + type: string + label: Active Directory resource ID + required: false + group: "Azure Configuration" + +# ======================== +# Authentication +# ======================== + +- variable: auth.basicAuth.enabled + description: "Configures basic authentication for the K10 dashboard" + type: boolean + label: Enable Basic Authentication + required: false + group: "Authentication" + show_subquestion_if: true + subquestions: + - variable: auth.basicAuth.htpasswd + description: "A username and password pair separated by a colon character" + type: password + label: Authentication Details (htpasswd) + - variable: auth.basicAuth.secretName + description: "Name of an existing Secret that contains a file generated with htpasswd" + type: string + label: Secret Name + +- variable: auth.tokenAuth.enabled + description: "Configures token based authentication for the K10 dashboard" + type: boolean + label: Enable Token Based Authentication + required: false + group: "Authentication" + +- variable: auth.oidcAuth.enabled + description: "Configures Open ID Connect based authentication for the K10 dashboard" + type: boolean + label: Enable OpenID Connect Based Authentication + required: false + group: "Authentication" + show_subquestion_if: true + subquestions: + - variable: auth.oidcAuth.providerURL + description: "URL for the OIDC Provider" + type: string + label: OIDC Provider URL + - variable: auth.oidcAuth.redirectURL + description: "URL for the K10 gateway Provider" + type: string + label: OIDC Redirect URL + - variable: auth.oidcAuth.scopes + description: "Space separated OIDC scopes required for userinfo. Example: `profile email`" + type: string + label: OIDC scopes + - variable: auth.oidcAuth.prompt + description: "The type of prompt to be used during authentication (none, consent, login, or select_account)" + type: enum + options: + - none + - consent + - login + - select_account + default: none + label: The type of prompt to be used during authentication (none, consent, login, or select_account) + - variable: auth.oidcAuth.clientID + description: "Client ID given by the OIDC provider for K10" + type: password + label: OIDC Client ID + - variable: auth.oidcAuth.clientSecret + description: "Client secret given by the OIDC provider for K10" + type: password + label: OIDC Client Secret + - variable: auth.oidcAuth.usernameClaim + description: "The claim to be used as the username" + type: string + label: OIDC UserName Claim + - variable: auth.oidcAuth.usernamePrefix + description: "Prefix that has to be used with the username obtained from the username claim" + type: string + label: OIDC UserName Prefix + - variable: auth.oidcAuth.groupClaim + description: "Name of a custom OpenID Connect claim for specifying user groups" + type: string + label: OIDC group Claim + - variable: auth.oidcAuth.groupPrefix + description: "All groups will be prefixed with this value to prevent conflicts" + type: string + label: OIDC group Prefix + +# ======================== +# External Gateway +# ======================== + +- variable: externalGateway.create + description: "Configures an external gateway for K10 API services" + type: boolean + label: Create External Gateway + required: false + group: "External Gateway" + show_subquestion_if: true + subquestions: + - variable: externalGateway.annotations + description: "Standard annotations for the services" + type: multiline + default: "" + label: Annotation + - variable: externalGateway.fqdn.name + description: "Domain name for the K10 API services" + type: string + label: Domain Name + - variable: externalGateway.fqdn.type + description: "Supported gateway type: `route53-mapper` or `external-dns`" + type: string + label: Gateway Type route53-mapper or external-dns + - variable: externalGateway.awsSSLCertARN + description: "ARN for the AWS ACM SSL certificate used in the K10 API server" + type: multiline + label: ARN for the AWS ACM SSL certificate + +# ======================== +# Storage Management +# ======================== + +- variable: global.persistence.storageClass + label: StorageClass Name + description: "Specifies StorageClass Name to be used for PVCs" + type: string + required: false + default: "" + group: "Storage Management" + +- variable: prometheus.server.persistentVolume.storageClass + type: string + label: StorageClass Name for Prometheus PVC + description: "StorageClassName used to create Prometheus PVC. Setting this option overwrites global StorageClass value" + default: "" + required: false + group: "Storage Management" + +- variable: prometheus.server.persistentVolume.enabled + type: boolean + label: Enable PVC for Prometheus server + description: "If true, K10 Prometheus server will create a Persistent Volume Claim" + default: true + required: false + group: "Storage Management" + +- variable: global.persistence.enabled + type: boolean + label: Storage Enabled + description: "If true, K10 will use Persistent Volume Claim" + default: true + required: false + group: "Storage Management" + +# ======================== +# Service Account +# ======================== + +- variable: serviceAccount.name + description: "Name of a service account in the target namespace that has cluster-admin permissions. This is needed for the K10 to be able to protect cluster resources." + type: string + label: Service Account Name + required: false + group: "Service Account" + +# ======================== +# License +# ======================== + +- variable: license + description: "License string obtained from Kasten" + type: multiline + label: License String + group: "License" +- variable: eula.accept + description: "Whether to enable accept EULA before installation" + type: boolean + label: Enable accept EULA before installation + group: "License" + show_subquestion_if: true + subquestions: + - variable: eula.company + description: "Company name. Required field if EULA is accepted" + type: string + label: Company Name + - variable: eula.email + description: "Contact email. Required field if EULA is accepted" + type: string + label: Contact Email diff --git a/charts/kasten/k10/7.0.1401/templates/NOTES.txt b/charts/kasten/k10/7.0.1401/templates/NOTES.txt new file mode 100644 index 000000000..b07806ce2 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/NOTES.txt @@ -0,0 +1,222 @@ +Thank you for installing Kasten’s K10 Data Management Platform {{ .Chart.Version }}! +{{- if .Values.fips.enabled }} + +You are operating in FIPS mode. +{{- end }} + +Documentation can be found at https://docs.kasten.io/. + +How to access the K10 Dashboard: + +{{- if .Values.ingress.create }} + +You are using the system's default ingress controller. Please ask your +administrator for instructions on how to access the cluster. + +WebUI location: https://{{ default "Your ingress endpoint" .Values.ingress.host }}/{{ default .Release.Name .Values.ingress.urlPath }} + +In addition, +{{- end }} + +{{- if .Values.route.enabled }} +WebUI location: https://{{ default "k10-route endpoint" .Values.route.host}}/{{ default .Release.Name .Values.route.path }}/ + +In addition, +{{- end }} + +{{- if .Values.externalGateway.create }} +{{- if .Values.externalGateway.fqdn.name }} + +The K10 Dashboard is accessible via {{ if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) .Values.externalGateway.awsSSLCertARN }}https{{ else }}http{{ end }}://{{ .Values.externalGateway.fqdn.name }}/{{ .Release.Name }}/#/ + +In addition, +{{- else }} + +The K10 Dashboard is accessible via a LoadBalancer. Find the service's EXTERNAL IP using: + `kubectl get svc gateway-ext --namespace {{ .Release.Namespace }} -o wide` +And use it in following URL + `http://SERVICE_EXTERNAL_IP/{{ .Release.Name }}/#/` + +In addition, +{{- end }} +{{- end }} + +To establish a connection to it use the following `kubectl` command: + +`kubectl --namespace {{ .Release.Namespace }} port-forward service/gateway 8080:{{ .Values.gateway.service.externalPort }}` + +The Kasten dashboard will be available at: `http{{ if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) .Values.externalGateway.awsSSLCertARN }}s{{ end }}://127.0.0.1:8080/{{ .Release.Name }}/#/` +{{ if and ( .Values.metering.awsManagedLicense ) ( not .Values.metering.licenseConfigSecretName ) }} + +IAM Role created during installation need to have permissions that allow K10 to +perform operations on EBS and, if needed, EFS and S3. Please create a policy +with required permissions, and use the commands below to attach the policy to +the service account. + +`ROLE_NAME=$(kubectl get serviceaccount {{ .Values.serviceAccount.name }} -n {{ .Release.Namespace }} -ojsonpath="{.metadata.annotations['eks\.amazonaws\.com/role-arn']}" | awk -F '/' '{ print $(NF) }')` +`aws iam attach-role-policy --role-name "${ROLE_NAME}" --policy-arn ` + +Refer to `https://docs.kasten.io/latest/install/aws-containers-anywhere/aws-containers-anywhere.html#attaching-permissions-for-eks-installations` +for more information. + +{{ end }} + +{{- if .Values.restore }} +{{- if or (empty .Values.restore.copyImagePullSecrets) (.Values.restore.copyImagePullSecrets) }} +-------------------- +Removal warning: The helm field `restore.copyImagePullSecrets` has been removed in version 6.0.12. K10 no longer copies the `imagePullSecret` to the application namespace. +-------------------- +{{- end }} +{{- end }} + +{{- if or (not (empty .Values.garbagecollector.importRunActions)) (not (empty .Values.garbagecollector.backupRunActions)) (not (empty .Values.garbagecollector.retireActions)) }} +Deprecation warning: The `garbagecollector.importRunActions`, `garbagecollector.backupRunActions`, `garbagecollector.retireActions` +blocks within the helm chart values have been replaced with `garbagecollector.actions`. +{{- end }} + +{{- if .Values.secrets.azureADEndpoint }} +-------------------- +Deprecation warning: The helm field `secret.azureADEndpoint` is deprecated and will be removed in upcoming release, we recommend you to use correct respective field, i.e., `secrets.microsoftEntraIDEndpoint`. +-------------------- +{{- end }} + + +{{- if .Values.secrets.azureADResourceID }} +-------------------- +Deprecation warning: The helm field `secret.azureADResourceID` is deprecated and will be removed in upcoming release, we recommend you to use correct respective field, i.e., `secrets.microsoftEntraIDResourceID` +-------------------- +{{- end }} + +{{- if .Values.grafana.enabled }} +-------------------- +Deprecation warning: Grafana will no longer be included in the Veeam Kasten installation process from the upcoming release 7.5.0. Upon upgrading to this (7.5.0) version, the integrated version of Grafana will be removed. It is important to install Grafana separately and follow the procedure described in our knowledge base article (https://www.veeam.com/kb4635) to configure the Kasten dashboards and alerts before upgrading Kasten to version 7.5.0. +-------------------- +{{- end }} + +{{- if or .Values.kanisterPodCustomLabels .Values.kanisterPodCustomAnnotations }} +-------------------- +Deprecation warning: The Helm values `kanisterPodCustomLabels` and `kanisterPodCustomAnnotations` are deprecated and will be removed in an upcoming release. Please use `global.podLabels` and `global.podAnnotations` to set labels and annotations to all the Kasten pods globally. +-------------------- +{{- end }} + +{{- if or .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey }} +-------------------- +Deprecation warning: The Helm values `secrets.apiTlsCrt` and `secrets.apiTlsKey` are deprecated and will be removed in an upcoming release. Please use `secrets.tlsSecret` to specify the name of a secret of type `kubernetes.io/tls`. This reduces the security risk of caching the certificates and keys in the shell history. +-------------------- +{{- end }} + +{{- if .Values.executorReplicas }} +-------------------- +Deprecation warning: The Helm value 'executorReplicas' is deprecated and will be removed in an upcoming release. Please use 'limiter.executorReplicas' instead. +-------------------- +{{- end }} +{{- if .Values.kanisterPodMetricSidecar.enabled }} +-------------------- +Deprecation warning: The Helm value 'kanisterPodMetricSidecar.enabled' is deprecated and will be removed in an upcoming release. Please use 'workerPodMetricSidecar.enabled' instead. +-------------------- +{{- end }} +{{- if .Values.services.executor.workerCount }} +-------------------- +Deprecation warning: The Helm value 'services.executor.workerCount' is deprecated and will be removed in an upcoming release. Please use 'limiter.executorThreads' instead. +-------------------- +{{- end }} +{{- if .Values.services.executor.maxConcurrentRestoreCsiSnapshots }} +-------------------- +Deprecation warning: The Helm value 'services.executor.maxConcurrentRestoreCsiSnapshots' is deprecated and will be removed in an upcoming release. Please use 'limiter.csiSnapshotRestoresPerAction' instead. +-------------------- +{{- end }} +{{- if .Values.services.executor.maxConcurrentRestoreGenericVolumeSnapshots }} +-------------------- +Deprecation warning: The Helm value 'services.executor.maxConcurrentRestoreGenericVolumeSnapshots' is deprecated and will be removed in an upcoming release. Please use 'limiter.volumeRestoresPerAction' instead. +-------------------- +{{- end }} +{{- if .Values.services.executor.maxConcurrentRestoreWorkloads }} +-------------------- +Deprecation warning: The Helm value 'services.executor.maxConcurrentRestoreWorkloads' is deprecated and will be removed in an upcoming release. Please use 'limiter.workloadRestoresPerAction' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.backupTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.backupTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.blueprintBackup' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.restoreTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.restoreTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.blueprintRestore' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.deleteTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.deleteTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.blueprintDelete' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.hookTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.hookTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.blueprintHooks' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.checkRepoTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.checkRepoTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.checkRepoPodReady' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.statsTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.statsTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.statsPodReady' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.efsPostRestoreTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.efsPostRestoreTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.efsRestorePodReady' instead. +-------------------- +{{- end }} +{{- if .Values.kanister.podReadyWaitTimeout }} +-------------------- +Deprecation warning: The Helm value 'kanister.podReadyWaitTimeout' is deprecated and will be removed in an upcoming release. Please use 'timeout.workerPodReady' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.concurrentSnapConversions }} +-------------------- +Deprecation warning: The Helm value 'limiter.concurrentSnapConversions' is deprecated and will be removed in an upcoming release. Please use 'limiter.snapshotExportsPerAction' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.genericVolumeSnapshots }} +-------------------- +Deprecation warning: The Helm value 'limiter.genericVolumeSnapshots' is deprecated and will be removed in an upcoming release. Please use 'limiter.genericVolumeBackupsPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.genericVolumeCopies }} +-------------------- +Deprecation warning: The Helm value 'limiter.genericVolumeCopies' is deprecated and will be removed in an upcoming release. Please use 'limiter.snapshotExportsPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.genericVolumeRestores }} +-------------------- +Deprecation warning: The Helm value 'limiter.genericVolumeRestores' is deprecated and will be removed in an upcoming release. Please use 'limiter.volumeRestoresPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.csiSnapshots }} +-------------------- +Deprecation warning: The Helm value 'limiter.csiSnapshots' is deprecated and will be removed in an upcoming release. Please use 'limiter.csiSnapshotsPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.providerSnapshots }} +-------------------- +Deprecation warning: The Helm value 'limiter.providerSnapshots' is deprecated and will be removed in an upcoming release. Please use 'limiter.directSnapshotsPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.limiter.imageCopies }} +-------------------- +Deprecation warning: The Helm value 'limiter.imageCopies' is deprecated and will be removed in an upcoming release. Please use 'limiter.imageCopiesPerCluster' instead. +-------------------- +{{- end }} +{{- if .Values.maxJobWaitDuration }} +-------------------- +Deprecation warning: The Helm value 'maxJobWaitDuration' is deprecated and will be removed in an upcoming release. Please use 'timeout.jobWait' instead. +-------------------- +{{- end }} +{{- if .Values.forceRootInKanisterHooks }} +-------------------- +Deprecation warning: The Helm value 'forceRootInKanisterHooks' is deprecated and will be removed in an upcoming release. Please use 'forceRootInBlueprintActions' instead. +-------------------- +{{- end }} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/templates/_definitions.tpl b/charts/kasten/k10/7.0.1401/templates/_definitions.tpl new file mode 100644 index 000000000..a10ca3b4e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_definitions.tpl @@ -0,0 +1,256 @@ +{{/* Code generated automatically. DO NOT EDIT. */}} +{{/* K10 services can be disabled by customers via helm value based feature flags. +Therefore, fetching of a list or yaml with service names should be done with the get.enabled* helper functions. +For example, the k10.restServices list can be fetched with get.enabledRestServices */}} +{{- define "k10.additionalServices" -}}frontend kanister{{- end -}} +{{- define "k10.restServices" -}}auth bloblifecyclemanager catalog controllermanager crypto dashboardbff events executor garbagecollector jobs logging metering repositories state vbrintegrationapi{{- end -}} +{{- define "k10.services" -}}aggregatedapis gateway{{- end -}} +{{- define "k10.exposedServices" -}}auth dashboardbff vbrintegrationapi{{- end -}} +{{- define "k10.statelessServices" -}}aggregatedapis auth bloblifecyclemanager controllermanager crypto dashboardbff events executor garbagecollector repositories gateway state vbrintegrationapi{{- end -}} +{{- define "k10.colocatedServices" -}} +bloblifecyclemanager: + port: 8001 + primary: crypto +events: + port: 8001 + primary: state +garbagecollector: + port: 8002 + primary: crypto +repositories: + port: 8003 + primary: crypto +vbrintegrationapi: + port: 8001 + primary: dashboardbff +{{- end -}} +{{- define "k10.colocatedServiceLookup" -}} +crypto: +- bloblifecyclemanager +- garbagecollector +- repositories +dashboardbff: +- vbrintegrationapi +state: +- events +{{- end -}} +{{- define "k10.aggregatedAPIs" -}}actions apps repositories vault dr{{- end -}} +{{- define "k10.configAPIs" -}}config{{- end -}} +{{- define "k10.profiles" -}}profiles{{- end -}} +{{- define "k10.policies" -}}policies{{- end -}} +{{- define "k10.policypresets" -}}policypresets{{- end -}} +{{- define "k10.transformsets" -}}transformsets{{- end -}} +{{- define "k10.blueprintbindings" -}}blueprintbindings{{- end -}} +{{- define "k10.auditconfigs" -}}auditconfigs{{- end -}} +{{- define "k10.storagesecuritycontexts" -}}storagesecuritycontexts{{- end -}} +{{- define "k10.storagesecuritycontextbindings" -}}storagesecuritycontextbindings{{- end -}} +{{- define "k10.reportingAPIs" -}}reporting{{- end -}} +{{- define "k10.distAPIs" -}}dist{{- end -}} +{{- define "k10.actionsAPIs" -}}actions{{- end -}} +{{- define "k10.backupActions" -}}backupactions{{- end -}} +{{- define "k10.backupActionsDetails" -}}backupactions/details{{- end -}} +{{- define "k10.reportActions" -}}reportactions{{- end -}} +{{- define "k10.reportActionsDetails" -}}reportactions/details{{- end -}} +{{- define "k10.storageRepositories" -}}storagerepositories{{- end -}} +{{- define "k10.restoreActions" -}}restoreactions{{- end -}} +{{- define "k10.restoreActionsDetails" -}}restoreactions/details{{- end -}} +{{- define "k10.importActions" -}}importactions{{- end -}} +{{- define "k10.exportActions" -}}exportactions{{- end -}} +{{- define "k10.exportActionsDetails" -}}exportactions/details{{- end -}} +{{- define "k10.retireActions" -}}retireactions{{- end -}} +{{- define "k10.runActions" -}}runactions{{- end -}} +{{- define "k10.runActionsDetails" -}}runactions/details{{- end -}} +{{- define "k10.backupClusterActions" -}}backupclusteractions{{- end -}} +{{- define "k10.backupClusterActionsDetails" -}}backupclusteractions/details{{- end -}} +{{- define "k10.restoreClusterActions" -}}restoreclusteractions{{- end -}} +{{- define "k10.restoreClusterActionsDetails" -}}restoreclusteractions/details{{- end -}} +{{- define "k10.cancelActions" -}}cancelactions{{- end -}} +{{- define "k10.upgradeActions" -}}upgradeactions{{- end -}} +{{- define "k10.appsAPIs" -}}apps{{- end -}} +{{- define "k10.restorePoints" -}}restorepoints{{- end -}} +{{- define "k10.restorePointsDetails" -}}restorepoints/details{{- end -}} +{{- define "k10.clusterRestorePoints" -}}clusterrestorepoints{{- end -}} +{{- define "k10.clusterRestorePointsDetails" -}}clusterrestorepoints/details{{- end -}} +{{- define "k10.applications" -}}applications{{- end -}} +{{- define "k10.applicationsDetails" -}}applications/details{{- end -}} +{{- define "k10.vaultAPIs" -}}vault{{- end -}} +{{- define "k10.passkey" -}}passkeys{{- end -}} +{{- define "k10.authAPIs" -}}auth{{- end -}} +{{- define "k10.defaultK10LimiterSnapshotExportsPerAction" -}}3{{- end -}} +{{- define "k10.defaultK10LimiterWorkloadSnapshotsPerAction" -}}5{{- end -}} +{{- define "k10.defaultK10DataStoreParallelUpload" -}}8{{- end -}} +{{- define "k10.defaultK10DataStoreGeneralContentCacheSizeMB" -}}0{{- end -}} +{{- define "k10.defaultK10DataStoreGeneralMetadataCacheSizeMB" -}}500{{- end -}} +{{- define "k10.defaultK10DataStoreRestoreContentCacheSizeMB" -}}500{{- end -}} +{{- define "k10.defaultK10DataStoreRestoreMetadataCacheSizeMB" -}}500{{- end -}} +{{- define "k10.defaultK10BackupBufferFileHeadroomFactor" -}}1.1{{- end -}} +{{- define "k10.defaultK10LimiterGenericVolumeBackupsPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterSnapshotExportsPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterVolumeRestoresPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterCsiSnapshotsPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterImageCopiesPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterDirectSnapshotsPerCluster" -}}10{{- end -}} +{{- define "k10.defaultK10GCDaemonPeriod" -}}21600{{- end -}} +{{- define "k10.defaultK10GCKeepMaxActions" -}}1000{{- end -}} +{{- define "k10.defaultK10GCActionsEnabled" -}}false{{- end -}} +{{- define "k10.defaultK10LimiterExecutorThreads" -}}8{{- end -}} +{{- define "k10.defaultK10LimiterCsiSnapshotRestoresPerAction" -}}3{{- end -}} +{{- define "k10.defaultK10LimiterVolumeRestoresPerAction" -}}3{{- end -}} +{{- define "k10.defaultK10LimiterWorkloadRestoresPerAction" -}}3{{- end -}} +{{- define "k10.defaultAssumeRoleDuration" -}}60m{{- end -}} +{{- define "k10.defaultK10TimeoutBlueprintBackup" -}}45{{- end -}} +{{- define "k10.defaultK10TimeoutBlueprintRestore" -}}600{{- end -}} +{{- define "k10.defaultK10TimeoutBlueprintDelete" -}}45{{- end -}} +{{- define "k10.defaultK10TimeoutBlueprintHooks" -}}20{{- end -}} +{{- define "k10.defaultK10TimeoutCheckRepoPodReady" -}}20{{- end -}} +{{- define "k10.defaultK10TimeoutStatsPodReady" -}}20{{- end -}} +{{- define "k10.defaultK10TimeoutEFSRestorePodReady" -}}45{{- end -}} +{{- define "k10.cloudProviders" -}}aws google azure{{- end -}} +{{- define "k10.serviceResources" -}} +aggregatedapis-svc: + aggregatedapis-svc: + requests: + cpu: 90m + memory: 180Mi +auth-svc: + auth-svc: + requests: + cpu: 2m + memory: 30Mi +catalog-svc: + catalog-svc: + requests: + cpu: 200m + memory: 780Mi + kanister-sidecar: + limits: + cpu: 1200m + memory: 800Mi + requests: + cpu: 100m + memory: 800Mi + schema-upgrade-check: + requests: + cpu: 200m + memory: 780Mi + upgrade-init: + requests: + cpu: 5m + memory: 20Mi +controllermanager-svc: + controllermanager-svc: + requests: + cpu: 5m + memory: 30Mi +crypto-svc: + bloblifecyclemanager-svc: + requests: + cpu: 10m + memory: 40Mi + crypto-svc: + requests: + cpu: 1m + memory: 30Mi + garbagecollector-svc: + requests: + cpu: 3m + memory: 100Mi + repositories-svc: + requests: + cpu: 10m + memory: 40Mi +dashboardbff-svc: + dashboardbff-svc: + requests: + cpu: 8m + memory: 40Mi + vbrintegrationapi-svc: + requests: + cpu: 3m + memory: 120Mi +executor-svc: + executor-svc: + requests: + cpu: 3m + memory: 50Mi + tools: + requests: + cpu: 1m + memory: 2Mi +frontend-svc: + frontend-svc: + requests: + cpu: 1m + memory: 40Mi +jobs-svc: + jobs-svc: + requests: + cpu: 30m + memory: 380Mi + upgrade-init: + requests: + cpu: 5m + memory: 20Mi +kanister-svc: + kanister-svc: + requests: + cpu: 1m + memory: 30Mi +logging-svc: + logging-svc: + requests: + cpu: 2m + memory: 40Mi + upgrade-init: + requests: + cpu: 5m + memory: 20Mi +metering-svc: + metering-svc: + requests: + cpu: 2m + memory: 30Mi + upgrade-init: + requests: + cpu: 5m + memory: 20Mi +state-svc: + events-svc: + requests: + cpu: 3m + memory: 500Mi + state-svc: + requests: + cpu: 2m + memory: 30Mi +{{- end -}} +{{- define "k10.multiClusterVersion" -}}2.5{{- end -}} +{{- define "k10.mcExternalPort" -}}18000{{- end -}} +{{- define "k10.defaultKubeVirtVMsUnfreezeTimeout" -}}5m{{- end -}} +{{- define "k10.aggAuditPolicyFile" -}}agg-audit-policy.yaml{{- end -}} +{{- define "k10.siemAuditLogFilePath" -}}-{{- end -}} +{{- define "k10.siemAuditLogFileSize" -}}100{{- end -}} +{{- define "k10.kanisterToolsImageTag" -}}0.112.0{{- end -}} +{{- define "k10.disabledServicesEnvVar" -}}K10_DISABLED_SERVICES{{- end -}} +{{- define "k10.openShiftClientSecretEnvVar" -}}K10_OPENSHIFT_CLIENT_SECRET{{- end -}} +{{- define "k10.defaultK10DefaultPriorityClassName" -}}{{- end -}} +{{- define "k10.dexServiceAccountName" -}}k10-dex-k10-sa{{- end -}} +{{- define "k10.defaultCACertConfigMapName" -}}custom-ca-bundle-store{{- end -}} +{{- define "k10.openShiftConsolePluginName" -}}veeam-kasten-console-plugin{{- end -}} +{{- define "k10.openShiftConsolePluginImageName" -}}ocpconsoleplugin{{- end -}} +{{- define "k10.gatewayPrefixVarName" -}}PREFIX_PATH{{- end -}} +{{- define "k10.gatewayGrafanaSvcVarName" -}}GRAFANA_SVC_NAME{{- end -}} +{{- define "k10.gatewayRequestHeadersVarName" -}}EXTAUTH_REQUEST_HEADERS{{- end -}} +{{- define "k10.gatewayAuthHeadersVarName" -}}EXTAUTH_AUTH_HEADERS{{- end -}} +{{- define "k10.gatewayPortVarName" -}}PORT{{- end -}} +{{- define "k10.gatewayEnableDex" -}}ENABLE_DEX{{- end -}} +{{- define "k10.gatewayTLSCertFile" -}}TLS_CRT_FILE{{- end -}} +{{- define "k10.gatewayTLSKeyFile" -}}TLS_KEY_FILE{{- end -}} +{{- define "k10.azureClientIDEnvVar" -}}AZURE_CLIENT_ID{{- end -}} +{{- define "k10.azureTenantIDEnvVar" -}}AZURE_TENANT_ID{{- end -}} +{{- define "k10.azureClientSecretEnvVar" -}}AZURE_CLIENT_SECRET{{- end -}} +{{- define "k10.oidcSecretName" -}}k10-oidc-auth{{- end -}} +{{- define "k10.oidcCustomerSecretName" -}}k10-oidc-auth-creds{{- end -}} +{{- define "k10.secretsDir" -}}/var/run/secrets/kasten.io{{- end -}} +{{- define "k10.sccNameEnvVar" -}}K10_SCC_NAME{{- end -}} +{{- define "k10.fluentbitEndpointEnvVar" -}}FLUENTBIT_ENDPOINT{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/_grafana.tpl b/charts/kasten/k10/7.0.1401/templates/_grafana.tpl new file mode 100644 index 000000000..53ade64a0 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_grafana.tpl @@ -0,0 +1,18 @@ +{{/*** SELECTOR LABELS *** + NOTE: The selector labels here (`app` and `release`) are divergent from + the selector labels set by the upstream chart. This is intentional since a + Deployment's `spec.selector` is immutable and K10 has already been shipped + with these values. + + A change to these selector labels will mean that all customers must manually + delete the Grafana Deployment before upgrading, which is a situation we don't + want for our customers. + + Instead, the `app.kubernetes.io/name` and `app.kubernetes.io/instance` labels + are included in the `grafana.extraLabels` in: + `templates/{values}/grafana/values/grafana_values.tpl`. +*/}} +{{- define "grafana.selectorLabels" -}} +app: {{ include "grafana.name" . }} +release: {{ .Release.Name }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/_helpers.tpl b/charts/kasten/k10/7.0.1401/templates/_helpers.tpl new file mode 100644 index 000000000..746622431 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_helpers.tpl @@ -0,0 +1,1614 @@ +{{/* Returns a string of the disabled K10 services */}} +{{- define "get.disabledServices" -}} + {{/* Append services to this list based on helm values */}} + {{- $disabledServices := list -}} + + {{- if eq .Values.logging.internal false -}} + {{- $disabledServices = append $disabledServices "logging" -}} + {{- end -}} + + {{- $disabledServices | join " " -}} +{{- end -}} + +{{/* Removes disabled service names from the provided string of service names */}} +{{- define "removeDisabledServicesFromList" -}} + {{- $disabledServices := include "get.disabledServices" .main | splitList " " -}} + {{- $services := .list | splitList " " -}} + + {{- range $disabledServices -}} + {{- $services = without $services . -}} + {{- end -}} + + {{- $services | join " " -}} +{{- end -}} + +{{/* Removes keys with disabled service names from the provided YAML string */}} +{{- define "removeDisabledServicesFromYaml" -}} + {{- $disabledServices := include "get.disabledServices" .main | splitList " " -}} + {{- $services := .yaml | fromYaml -}} + + {{- range $disabledServices -}} + {{- $services = unset $services . -}} + {{- end -}} + + {{- if gt (len $services) 0 -}} + {{- $services | toYaml | trim | nindent 0}} + {{- else -}} + {{- print "" -}} + {{- end -}} +{{- end -}} + +{{/* Returns k10.additionalServices string with disabled services removed */}} +{{- define "get.enabledAdditionalServices" -}} + {{- $list := include "k10.additionalServices" . -}} + {{- dict "main" . "list" $list | include "removeDisabledServicesFromList" -}} +{{- end -}} + +{{/* Returns k10.restServices string with disabled services removed */}} +{{- define "get.enabledRestServices" -}} + {{- $list := include "k10.restServices" . -}} + {{- dict "main" . "list" $list | include "removeDisabledServicesFromList" -}} +{{- end -}} + +{{/* Returns k10.services string with disabled services removed */}} +{{- define "get.enabledServices" -}} + {{- $list := include "k10.services" . -}} + {{- dict "main" . "list" $list | include "removeDisabledServicesFromList" -}} +{{- end -}} + +{{/* Returns k10.exposedServices string with disabled services removed */}} +{{- define "get.enabledExposedServices" -}} + {{- $list := include "k10.exposedServices" . -}} + {{- dict "main" . "list" $list | include "removeDisabledServicesFromList" -}} +{{- end -}} + +{{/* Returns k10.statelessServices string with disabled services removed */}} +{{- define "get.enabledStatelessServices" -}} + {{- $list := include "k10.statelessServices" . -}} + {{- dict "main" . "list" $list | include "removeDisabledServicesFromList" -}} +{{- end -}} + +{{/* Returns k10.colocatedServices string with disabled services removed */}} +{{- define "get.enabledColocatedServices" -}} + {{- $yaml := include "k10.colocatedServices" . -}} + {{- dict "main" . "yaml" $yaml | include "removeDisabledServicesFromYaml" -}} +{{- end -}} + +{{/* Returns YAML of primary services mapped to their secondary services */}} +{{/* The content will only have services which are not disabled */}} +{{- define "get.enabledColocatedServiceLookup" -}} + {{- $colocatedServicesLookup := include "k10.colocatedServiceLookup" . | fromYaml -}} + {{- $disabledServices := include "get.disabledServices" . | splitList " " -}} + {{- $filteredLookup := dict -}} + + {{/* construct filtered lookup */}} + {{- range $primaryService, $secondaryServices := $colocatedServicesLookup -}} + {{/* proceed only if primary service is enabled */}} + {{- if not (has $primaryService $disabledServices) -}} + {{/* filter out secondary services */}} + {{- range $disabledServices -}} + {{- $secondaryServices = without $secondaryServices . -}} + {{- end -}} + {{/* add entry for primary service only if secondary services exist */}} + {{- if gt (len $secondaryServices) 0 -}} + {{- $filteredLookup = set $filteredLookup $primaryService $secondaryServices -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{/* return filtered lookup */}} + {{- if gt (len $filteredLookup) 0 -}} + {{- $filteredLookup | toYaml | trim | nindent 0 -}} + {{- else -}} + {{- print "" -}} + {{- end -}} +{{- end -}} + +{{- define "k10.capabilities" -}} + {{- /* Internal capabilities enabled by other Helm values are added here */ -}} + {{- $internal_capabilities := list "gateway" -}} + + {{- /* Multi-cluster */ -}} + {{- if eq .Values.multicluster.enabled true -}} + {{- $internal_capabilities = append $internal_capabilities "mc" -}} + {{- end -}} + + {{- /* FIPS */ -}} + {{- if .Values.fips.enabled -}} + {{- $internal_capabilities = append $internal_capabilities "fips.strict" -}} + {{- $internal_capabilities = append $internal_capabilities "crypto.k10.v2" -}} + {{- $internal_capabilities = append $internal_capabilities "crypto.storagerepository.v2" -}} + {{- $internal_capabilities = append $internal_capabilities "crypto.vbr.v2" -}} + {{- $internal_capabilities = append $internal_capabilities "gateway" -}} + {{- end -}} + + {{- concat $internal_capabilities (.Values.capabilities | default list) | join " " -}} +{{- end -}} + +{{- define "k10.capabilities_mask" -}} + {{- /* Internal capabilities masked by other Helm values are added here */ -}} + {{- $internal_capabilities_mask := list -}} + + {{- /* Multi-cluster */ -}} + {{- if eq .Values.multicluster.enabled false -}} + {{- $internal_capabilities_mask = append $internal_capabilities_mask "mc" -}} + {{- end -}} + + {{- concat $internal_capabilities_mask (.Values.capabilitiesMask | default list) | join " " -}} +{{- end -}} + +{{/* + k10.capability checks whether a given capability is enabled + + For example: + + include "k10.capability" (. | merge (dict "capability" "SOME.CAPABILITY")) +*/}} +{{- define "k10.capability" -}} + {{- $capabilities := dict -}} + {{- range $capability := include "k10.capabilities" . | splitList " " -}} + {{- $_ := set $capabilities $capability "enabled" -}} + {{- end -}} + {{- range $capability := include "k10.capabilities_mask" . | splitList " " -}} + {{- $_ := unset $capabilities $capability -}} + {{- end -}} + + {{- index $capabilities .capability | default "" -}} +{{- end -}} + +{{/* + k10.capability.gateway checks whether the "gateway" capability is enabled +*/}} +{{- define "k10.capability.gateway" -}} + {{- include "k10.capability" (. | merge (dict "capability" "gateway")) -}} +{{- end -}} + +{{/* Check if basic auth is needed */}} +{{- define "basicauth.check" -}} + {{- if .Values.auth.basicAuth.enabled }} + {{- print true }} + {{- end -}} {{/* End of check for auth.basicAuth.enabled */}} +{{- end -}} + +{{/* +Check if trusted root CA certificate related configmap settings +have been configured +*/}} +{{- define "check.cacertconfigmap" -}} +{{- if .Values.cacertconfigmap.name -}} +{{- print true -}} +{{- else -}} +{{- print false -}} +{{- end -}} +{{- end -}} + +{{/* +Check if OCP CA certificates automatic extraction is enabled +*/}} +{{- define "k10.ocpcacertsautoextraction" -}} + {{- if and .Values.auth.openshift.enabled .Values.auth.openshift.caCertsAutoExtraction -}} + {{- true -}} + {{- end -}} +{{- end -}} + +{{/* +Get the name of the CA certificate related configmap +*/}} +{{- define "k10.cacertconfigmapname" -}} + {{- if eq (include "check.cacertconfigmap" .) "true" -}} + {{- .Values.cacertconfigmap.name -}} + {{- else if (include "k10.ocpcacertsautoextraction" .) -}} + {{- include "k10.defaultCACertConfigMapName" . -}} + {{- end -}} +{{- end -}} + +{{- define "k10.sccAnnotations" -}} + {{- if .Values.scc.create -}} + {{- dict "openshift.io/required-scc" (printf "%s-scc" .Release.Name) | toYaml -}} + {{- end -}} +{{- end -}} + +{{/* Merging common pod labels for deployments with specific order of priority +to prevent overwriting of required labels. Certain site-specific required labels +are passed into the context as a dict at the caller site. */}} +{{- define "k10.deploymentPodLabels" -}} + {{- (merge (dict) (.requiredLabels) (include "k10.azMarketPlace.billingIdentifier" . | fromYaml) (include "helm.labels" . | fromYaml) (include "k10.globalPodLabels" . | fromYaml)) | toYaml -}} +{{- end -}} + +{{/* Merging common pod annotations for deployments with specific order of +priority to prevent overwriting of required annotations. Certain site-specific +required annotations are passed into the context as a dict at the caller site. */}} +{{- define "k10.deploymentPodAnnotations" -}} + {{- (merge (dict) (.requiredAnnotations) (dict "checksum/config" (include (print .Template.BasePath "/k10-config.yaml") . | sha256sum)) (dict "checksum/secret" (include (print .Template.BasePath "/secrets.yaml") . | sha256sum)) (include "k10.sccAnnotations" . | fromYaml) (include "k10.globalPodAnnotations" . | fromYaml)) | toYaml -}} +{{- end -}} + +{{/* Custom pod labels applied globally to all pods */}} +{{- define "k10.globalPodLabels" -}} + {{ include "k10.validateGlobalAndKanisterLabelsAnnotations" . }} + {{- with .Values.global.podLabels -}} + {{- toYaml . -}} + {{- end -}} +{{- end -}} + +{{/* Custom pod labels applied globally to all pods in a json format */}} +{{- define "k10.globalPodLabelsJson" -}} + {{- if .Values.global.podLabels -}} + {{- toJson .Values.global.podLabels -}} + {{- end -}} +{{- end -}} + +{{/* Custom pod annotations applied globally to all pods */}} +{{- define "k10.globalPodAnnotations" -}} + {{ include "k10.validateGlobalAndKanisterLabelsAnnotations" . }} + {{- with .Values.global.podAnnotations -}} + {{- toYaml . -}} + {{- end -}} +{{- end -}} + +{{/* Custom pod annotations applied globally to all pods in a json format */}} +{{- define "k10.globalPodAnnotationsJson" -}} + {{- if .Values.global.podAnnotations -}} + {{- toJson .Values.global.podAnnotations -}} + {{- end -}} +{{- end -}} + +{{/* +Validate and fail if the labels/annotations are configured at global level (global.podLabels) +as well as kanister helm field level (kanisterPodCustomLabels) +*/}} +{{- define "k10.validateGlobalAndKanisterLabelsAnnotations" -}} + {{- if and .Values.global.podAnnotations .Values.kanisterPodCustomAnnotations -}} + {{- fail "The `kanisterPodCustomAnnotations` field has been deprecated and cannot be used simultaneously with `global.podAnnotations`. Please use `global.podAnnotations` to set annotations to all the Kasten pods globally." }} + {{- end -}} + {{- if and .Values.global.podLabels .Values.kanisterPodCustomLabels -}} + {{- fail "The `kanisterPodCustomLabels` field has been deprecated and cannot be used simultaneously with `global.podLabels`. Please use `global.podLabels` to set labels to all the Kasten pods globally." }} + {{- end -}} +{{- end -}} + +{{/* +Check if the auth options are implemented using Dex +*/}} +{{- define "check.dexAuth" -}} +{{- if or .Values.auth.openshift.enabled .Values.auth.ldap.enabled -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* Check the only 1 auth is specified */}} +{{- define "singleAuth.check" -}} +{{- $count := dict "count" (int 0) -}} +{{- $authList := list .Values.auth.basicAuth.enabled .Values.auth.tokenAuth.enabled .Values.auth.oidcAuth.enabled .Values.auth.openshift.enabled .Values.auth.ldap.enabled -}} +{{- range $i, $val := $authList }} +{{ if $val }} +{{ $c := add1 $count.count | set $count "count" }} +{{ if gt $count.count 1 }} +{{- fail "Multiple auth types were selected. Only one type can be enabled." }} +{{ end }} +{{ end }} +{{- end }} +{{- end -}}{{/* Check the only 1 auth is specified */}} + +{{/* Check if Auth is enabled */}} +{{- define "authEnabled.check" -}} +{{- $count := dict "count" (int 0) -}} +{{- $authList := list .Values.auth.basicAuth.enabled .Values.auth.tokenAuth.enabled .Values.auth.oidcAuth.enabled .Values.auth.openshift.enabled .Values.auth.ldap.enabled -}} +{{- range $i, $val := $authList }} +{{ if $val }} +{{ $c := add1 $count.count | set $count "count" }} +{{ end }} +{{- end }} +{{- if eq $count.count 0}} + {{- fail "Auth is required to expose access to K10." }} +{{- end }} +{{- end -}}{{/*end of check */}} + +{{/* Return ingress class name annotation */}} +{{- define "ingressClassAnnotation" -}} +{{- if .Values.ingress.class -}} +kubernetes.io/ingress.class: {{ .Values.ingress.class | quote }} +{{- end -}} +{{- end -}} + +{{/* Return ingress class name in spec */}} +{{- define "specIngressClassName" -}} +{{- if and .Values.ingress.class (semverCompare ">= 1.27-0" .Capabilities.KubeVersion.Version) -}} +ingressClassName: {{ .Values.ingress.class }} +{{- end -}} +{{- end -}} + +{{/* Helm required labels */}} +{{- define "helm.labels" -}} +heritage: {{ .Release.Service }} +helm.sh/chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} +app.kubernetes.io/name: {{ .Chart.Name }} +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{ include "k10.common.matchLabels" . }} +{{- end -}} + +{{- define "k10.common.matchLabels" -}} +app: {{ .Chart.Name }} +release: {{ .Release.Name }} +{{- end -}} + +{{- define "k10.defaultRBACLabels" -}} +k10.kasten.io/default-rbac-object: "true" +{{- end -}} + +{{/* Expand the name of the chart. */}} +{{- define "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). +*/}} +{{- define "fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account to use +*/}} +{{- define "serviceAccountName" -}} +{{- if and .Values.metering.awsMarketplace ( not .Values.serviceAccount.name ) -}} + {{ print "k10-metering" }} +{{- else if .Values.serviceAccount.create -}} + {{ default (include "fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the name of the metering service account to use +*/}} +{{- define "meteringServiceAccountName" -}} +{{- if and .Values.metering.awsManagedLicense ( not .Values.serviceAccount.name ) ( not .Values.metering.serviceAccount.name ) ( not .Values.metering.licenseConfigSecretName ) -}} + {{ print "k10-metering" }} +{{- else -}} + {{ default (include "serviceAccountName" .) .Values.metering.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Prints annotations based on .Values.fqdn.type +*/}} +{{- define "dnsAnnotations" -}} +{{- if .Values.externalGateway.fqdn.name -}} +{{- if eq "route53-mapper" ( default "" .Values.externalGateway.fqdn.type) }} +domainName: {{ .Values.externalGateway.fqdn.name | quote }} +{{- end }} +{{- if eq "external-dns" (default "" .Values.externalGateway.fqdn.type) }} +external-dns.alpha.kubernetes.io/hostname: {{ .Values.externalGateway.fqdn.name | quote }} +{{- end }} +{{- end -}} +{{- end -}} + +{{/* +Prometheus scrape config template for k10 services +*/}} +{{- define "k10.prometheusScrape" -}} +{{- $cluster_domain := "" -}} +{{- with .main.Values.cluster.domainName -}} + {{- $cluster_domain = printf ".%s" . -}} +{{- end -}} +{{- $admin_port := default 8877 .main.Values.service.gatewayAdminPort -}} +- job_name: {{ .k10service }} + metrics_path: /metrics + {{- if eq "aggregatedapis" .k10service }} + scheme: https + tls_config: + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + {{- else }} + scheme: http + {{- end }} + static_configs: + - targets: + {{- if eq "gateway" .k10service }} + - {{ .k10service }}-admin.{{ .main.Release.Namespace }}.svc{{ $cluster_domain }}:{{ $admin_port }} + {{- else if eq "aggregatedapis" .k10service }} + - {{ .k10service }}-svc.{{ .main.Release.Namespace }}.svc{{ $cluster_domain }}:443 + {{- else }} + {{- $service := default .k10service (index (include "get.enabledColocatedServices" . | fromYaml) .k10service).primary }} + {{- $port := default .main.Values.service.externalPort (index (include "get.enabledColocatedServices" . | fromYaml) .k10service).port }} + - {{ $service }}-svc.{{ .main.Release.Namespace }}.svc{{ $cluster_domain }}:{{ $port }} + {{- end }} + labels: + application: {{ .main.Release.Name }} + service: {{ .k10service }} +{{- end -}} + +{{/* +Prometheus scrape config template for k10 services +*/}} +{{- define "k10.prometheusTargetConfig" -}} +{{- $cluster_domain := "" -}} +{{- with .main.Values.cluster.domainName -}} + {{- $cluster_domain = printf ".%s" . -}} +{{- end -}} +{{- $admin_port := default 8877 .main.Values.service.gatewayAdminPort | toString -}} +- service: {{ .k10service }} + metricsPath: /metrics + {{- if eq "aggregatedapis" .k10service }} + scheme: https + tls_config: + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + {{- else }} + scheme: http + {{- end }} + {{- $serviceFqdn := "" }} + {{- $servicePort := "" }} + {{- if eq "gateway" .k10service -}} + {{- $serviceFqdn = printf "%s-admin.%s.svc%s" .k10service .main.Release.Namespace $cluster_domain -}} + {{- $servicePort = $admin_port -}} + {{- else if eq "aggregatedapis" .k10service -}} + {{- $serviceFqdn = printf "%s-svc.%s.svc%s" .k10service .main.Release.Namespace $cluster_domain -}} + {{- $servicePort = "443" -}} + {{- else -}} + {{- $service := default .k10service (index (include "get.enabledColocatedServices" .main | fromYaml) .k10service).primary -}} + {{- $port := default .main.Values.service.externalPort (index (include "get.enabledColocatedServices" .main | fromYaml) .k10service).port | toString -}} + {{- $serviceFqdn = printf "%s-svc.%s.svc%s" $service .main.Release.Namespace $cluster_domain -}} + {{- $servicePort = $port -}} + {{- end }} + fqdn: {{ $serviceFqdn }} + port: {{ $servicePort }} + application: {{ .main.Release.Name }} +{{- end -}} + +{{/* +Expands the name of the Prometheus chart. It is equivalent to what the +"prometheus.name" template does. It is needed because the referenced values in a +template are relative to where/when the template is called from, and not where +the template is defined at. This means that the value of .Chart.Name and +.Values.nameOverride are different depending on whether the template is called +from within the Prometheus chart or the K10 chart. +*/}} +{{- define "k10.prometheus.name" -}} +{{- default "prometheus" .Values.prometheus.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Expands the name of the Prometheus service created to expose the prometheus server. +*/}} +{{- define "k10.prometheus.service.name" -}} +{{- default (printf "%s-%s-%s" .Release.Name "prometheus" .Values.prometheus.server.name) .Values.prometheus.server.fullnameOverride }} +{{- end -}} + +{{/* +Checks if EULA is accepted via cmd +Enforces eula.company and eula.email as required fields +returns configMap fields +*/}} +{{- define "k10.eula.fields" -}} +{{- if .Values.eula.accept -}} +accepted: "true" +company: {{ required "eula.company is required field if eula is accepted" .Values.eula.company }} +email: {{ required "eula.email is required field if eula is accepted" .Values.eula.email }} +{{- else -}} +accepted: "" +company: "" +email: "" +{{- end }} +{{- end -}} + +{{/* +Helper to determine the API Domain +*/}} +{{- define "apiDomain" -}} +{{- if .Values.useNamespacedAPI -}} +kio.{{- replace "-" "." .Release.Namespace -}} +{{- else -}} +kio.kasten.io +{{- end -}} +{{- end -}} + +{{/* +Get dex image, if user wants to +install certified version of upstream +images or not +*/}} + +{{- define "get.dexImage" }} + {{- (get .Values.global.images (include "dex.dexImageName" .)) | default (include "dex.dexImage" .) }} +{{- end }} + +{{- define "dex.dexImage" -}} + {{- printf "%s:%s" (include "dex.dexImageRepo" .) (include "dex.dexImageTag" .) }} +{{- end -}} + +{{- define "dex.dexImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "dex.dexImageName" .) }} + {{- else if .Values.global.azMarketPlace }} + {{- printf "%s/%s" .Values.global.azure.images.dex.registry .Values.global.azure.images.dex.image }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "dex.dexImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "dex.dexImageName" -}} + {{- printf "dex" }} +{{- end -}} + +{{- define "dex.dexImageTag" -}} + {{- if .Values.global.azMarketPlace }} + {{- print .Values.global.azure.images.dex.tag }} + {{- else }} + {{- .Values.global.image.tag | default .Chart.AppVersion }} + {{- end -}} +{{- end -}} + +{{/* + Get dex frontend directory (in the dex image) +*/}} +{{- define "k10.dexFrontendDir" -}} + {{- $dexImageDict := default $.Values.dexImage dict }} + {{- index $dexImageDict "frontendDir" | default "/srv/dex/web" }} +{{- end -}} + +{{/* +Get the k10tools image. +*/}} +{{- define "k10.k10ToolsImage" -}} + {{- (get .Values.global.images (include "k10.k10ToolsImageName" .)) | default (include "k10.k10ToolsDefaultImage" .) -}} +{{- end -}} + +{{- define "k10.k10ToolsDefaultImage" -}} + {{- printf "%s:%s" (include "k10.k10ToolsImageRepo" .) (include "k10.k10ToolsImageTag" .) -}} +{{- end -}} + +{{- define "k10.k10ToolsImageRepo" -}} + {{- if .Values.global.airgapped.repository -}} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "k10.k10ToolsImageName" .) -}} + {{- else if .Values.global.azMarketPlace -}} + {{- printf "%s/%s" .Values.global.azure.images.k10tools.registry .Values.global.azure.images.k10tools.image -}} + {{- else -}} + {{- printf "%s/%s" .Values.global.image.registry (include "k10.k10ToolsImageName" .) -}} + {{- end -}} +{{- end -}} + +{{- define "k10.k10ToolsImageName" -}} + {{- print "k10tools" -}} +{{- end -}} + +{{- define "k10.k10ToolsImageTag" -}} + {{- if .Values.global.azMarketPlace -}} + {{- print .Values.global.azure.images.k10tools.tag -}} + {{- else -}} + {{- include "get.k10ImageTag" . -}} + {{- end -}} +{{- end -}} + +{{/* +Get the ocpconsoleplugin image. +*/}} +{{- define "k10.ocpConsolePluginImage" -}} + {{- (get .Values.global.images (include "k10.openShiftConsolePluginImageName" .)) | default (include "k10.ocpConsolePluginDefaultImage" .) -}} +{{- end -}} + +{{- define "k10.ocpConsolePluginDefaultImage" -}} + {{- printf "%s:%s" (include "k10.ocpConsolePluginImageRepo" .) (include "get.k10ImageTag" .) -}} +{{- end -}} + +{{- define "k10.ocpConsolePluginImageRepo" -}} + {{- if .Values.global.airgapped.repository -}} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "k10.openShiftConsolePluginImageName" .) -}} + {{- else -}} + {{- printf "%s/%s" .Values.global.image.registry (include "k10.openShiftConsolePluginImageName" .) -}} + {{- end -}} +{{- end -}} + +{{/* +Get the emissary image. +*/}} +{{- define "get.emissaryImage" }} + {{- (get .Values.global.images (include "k10.emissaryImageName" .)) | default (include "k10.emissaryImage" .) }} +{{- end }} + +{{- define "k10.emissaryImage" -}} + {{- printf "%s:%s" (include "k10.emissaryImageRepo" .) (include "k10.emissaryImageTag" .) }} +{{- end -}} + +{{- define "k10.emissaryImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "k10.emissaryImageName" .) }} + {{- else if .Values.global.azMarketPlace }} + {{- printf "%s/%s" .Values.global.azure.images.emissary.registry .Values.global.azure.images.emissary.image }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "k10.emissaryImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "k10.emissaryImageName" -}} + {{- printf "emissary" }} +{{- end -}} + +{{- define "k10.emissaryImageTag" -}} + {{- if .Values.global.azMarketPlace }} + {{- print .Values.global.azure.images.emissary.tag }} + {{- else }} + {{- include "get.k10ImageTag" . }} + {{- end }} +{{- end -}} + +{{/* +Get the datamover image. +*/}} +{{- define "get.datamoverImage" }} + {{- (get .Values.global.images (include "k10.datamoverImageName" .)) | default (include "k10.datamoverImage" .) }} +{{- end }} + +{{- define "k10.datamoverImage" -}} + {{- printf "%s:%s" (include "k10.datamoverImageRepo" .) (include "k10.datamoverImageTag" .) }} +{{- end -}} + +{{- define "k10.datamoverImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "k10.datamoverImageName" .) }} + {{- else if .Values.global.azMarketPlace }} + {{- printf "%s/%s" .Values.global.azure.images.datamover.registry .Values.global.azure.images.datamover.image }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "k10.datamoverImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "k10.datamoverImageName" -}} + {{- printf "datamover" }} +{{- end -}} + +{{- define "k10.datamoverImageTag" -}} + {{- if .Values.global.azMarketPlace }} + {{- print .Values.global.azure.images.datamover.tag }} + {{- else }} + {{- include "get.k10ImageTag" . }} + {{- end }} +{{- end -}} + +{{/* +Get the metric-sidecar image. +*/}} +{{- define "get.metricSidecarImage" }} + {{- (get .Values.global.images (include "k10.metricSidecarImageName" .)) | default (include "k10.metricSidecarImage" .) }} +{{- end }} + +{{- define "k10.metricSidecarImage" -}} + {{- printf "%s:%s" (include "k10.metricSidecarImageRepo" .) (include "k10.metricSidecarImageTag" .) }} +{{- end -}} + +{{- define "k10.metricSidecarImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "k10.metricSidecarImageName" .) }} + {{- else if .Values.global.azMarketPlace }} + {{- printf "%s/%s" (.Values.global.azure.images.metricsidecar.registry) (.Values.global.azure.images.metricsidecar.image) }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "k10.metricSidecarImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "k10.metricSidecarImageName" -}} + {{- printf "metric-sidecar" }} +{{- end -}} + +{{- define "k10.metricSidecarImageTag" -}} + {{- if .Values.global.azMarketPlace }} + {{- print .Values.global.azure.images.metricsidecar.tag }} + {{- else }} + {{- include "get.k10ImageTag" . }} + {{- end }} +{{- end -}} + +{{/* +Check if AWS creds are specified +*/}} +{{- define "check.awscreds" -}} +{{- if or .Values.secrets.awsAccessKeyId .Values.secrets.awsSecretAccessKey -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.awsSecretName" -}} +{{- if .Values.secrets.awsClientSecretName -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.azureFederatedIdentity" -}} +{{- if and .Values.azure.useFederatedIdentity .Values.secrets.azureClientId -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Azure MSI with Default ID is specified +*/}} +{{- define "check.azureMSIWithDefaultID" -}} +{{- if .Values.azure.useDefaultMSI -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Azure MSI with a specific Client ID is specified +*/}} +{{- define "check.azureMSIWithClientID" -}} +{{- if and (not (or .Values.secrets.azureClientSecret .Values.secrets.azureTenantId)) .Values.secrets.azureClientId -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Azure ClientSecret creds are specified +*/}} +{{- define "check.azureClientSecretCreds" -}} +{{- if and (and .Values.secrets.azureTenantId .Values.secrets.azureClientId) .Values.secrets.azureClientSecret -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Checks and enforces only 1 set of azure creds is specified +*/}} +{{- define "enforce.singleazurecreds" -}} +{{- if and (eq (include "check.azureFederatedIdentity" .) "true") (eq (include "check.azureMSIWithDefaultID" .) "true") -}} +{{- fail "useDefaultMSI is set to true, but FederatedIdentity is also set to true. Please choose one." -}} +{{- end -}} +{{- if and (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureMSIWithDefaultID" .) "true") -}} +{{- fail "useDefaultMSI is set to true, but an additional ClientID is also provided. Please choose one." -}} +{{- end -}} +{{ if and ( or (eq (include "check.azureClientSecretCreds" .) "true") (eq (include "check.azuresecret" .) "true" )) (or (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureMSIWithDefaultID" .) "true")) }} +{{- fail "Both Azure ClientSecret and Managed Identity creds are available, but only one is allowed. Please choose one." }} +{{- end -}} +{{- end -}} + +{{/* +Get the kanister-tools image. +*/}} +{{- define "get.kanisterToolsImage" -}} + {{- (get .Values.global.images (include "kan.kanisterToolsImageName" .)) | default (include "kan.kanisterToolsImage" .) }} +{{- end }} + +{{- define "kan.kanisterToolsImage" -}} + {{- printf "%s:%s" (include "kan.kanisterToolsImageRepo" .) (include "kan.kanisterToolsImageTag" .) }} +{{- end -}} + +{{- define "kan.kanisterToolsImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "kan.kanisterToolsImageName" .) }} + {{- else if .Values.global.azMarketPlace }} + {{- printf "%s/%s" .Values.global.azure.images.kanistertools.registry .Values.global.azure.images.kanistertools.image }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "kan.kanisterToolsImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "kan.kanisterToolsImageName" -}} + {{- printf "kanister-tools" }} +{{- end -}} + +{{- define "kan.kanisterToolsImageTag" -}} + {{- if .Values.global.azMarketPlace }} + {{- print .Values.global.azure.images.kanistertools.tag }} + {{- else }} + {{- include "get.k10ImageTag" . }} + {{- end }} +{{- end -}} + +{{/* +Check if Google Workload Identity Federation is enabled +*/}} +{{- define "check.gwifenabled" -}} +{{- if .Values.google.workloadIdentityFederation.enabled -}} +{{- print true -}} +{{- end -}} +{{- end -}} + + +{{/* +Check if Google Workload Identity Federation Identity Provider is set +*/}} +{{- define "check.gwifidptype" -}} +{{- if .Values.google.workloadIdentityFederation.idp.type -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Fail if Google Workload Identity Federation is enabled but no Identity Provider is set +*/}} +{{- define "validate.gwif.idp.type" -}} +{{- if and (eq (include "check.gwifenabled" .) "true") (ne (include "check.gwifidptype" .) "true") -}} + {{- fail "Google Workload Federation is enabled but helm flag for idp type is missing. Please set helm value google.workloadIdentityFederation.idp.type" -}} +{{- end -}} +{{- end -}} + +{{/* +Check if K8S Bound Service Account Token (aka Projected Service Account Token) is needed, +which is when GWIF is enabled and the IdP is kubernetes +*/}} +{{- define "check.projectSAToken" -}} +{{- if and (eq (include "check.gwifenabled" .) "true") (eq .Values.google.workloadIdentityFederation.idp.type "kubernetes") -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if the audience that the bound service account token is intended for is set +*/}} +{{- define "check.gwifidpaud" -}} +{{- if .Values.google.workloadIdentityFederation.idp.aud -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Fail if Service Account token projection is expected but no indented Audience is set +*/}} +{{- define "validate.gwif.idp.aud" -}} +{{- if and (eq (include "check.projectSAToken" .) "true") (ne (include "check.gwifidpaud" .) "true") -}} + {{- fail "Kubernetes is set as the Identity Provider but an intended Audience is missing. Please set helm value google.workloadIdentityFederation.idp.aud" -}} +{{- end -}} +{{- end -}} + + +{{/* +Check if Google creds are specified +*/}} +{{- define "check.googlecreds" -}} +{{- if .Values.secrets.googleApiKey -}} + {{- if eq (include "check.isBase64" .Values.secrets.googleApiKey) "false" -}} + {{- fail "secrets.googleApiKey must be base64 encoded" -}} + {{- end -}} + {{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.googleCredsSecret" -}} +{{- if .Values.secrets.googleClientSecretName -}} + {{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.googleCredsOrSecret" -}} +{{- if or (eq (include "check.googlecreds" .) "true") (eq (include "check.googleCredsSecret" .) "true")}} + {{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Google Project ID is not set without Google API Key +*/}} +{{- define "check.googleproject" -}} +{{- if .Values.secrets.googleProjectId -}} + {{- if not .Values.secrets.googleApiKey -}} + {{- print false -}} + {{- else -}} + {{- print true -}} + {{- end -}} +{{- else -}} + {{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Azure creds are specified +*/}} +{{- define "check.azurecreds" -}} +{{- if or (eq (include "check.azureClientSecretCreds" .) "true") ( or (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureMSIWithDefaultID" .) "true")) -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.azuresecret" -}} +{{- if .Values.secrets.azureClientSecretName }} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Vsphere creds are specified +*/}} +{{- define "check.vspherecreds" -}} +{{- if or (or .Values.secrets.vsphereEndpoint .Values.secrets.vsphereUsername) .Values.secrets.vspherePassword -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.vsphereClientSecret" -}} +{{- if .Values.secrets.vsphereClientSecretName -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Vault token secret creds are specified +*/}} +{{- define "check.vaulttokenauth" -}} +{{- if .Values.vault.secretName -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if K8s role is specified +*/}} +{{- define "check.vaultk8sauth" -}} +{{- if .Values.vault.role -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{/* +Check if Vault creds for token or k8s auth are specified +*/}} +{{- define "check.vaultcreds" -}} +{{- if or (eq (include "check.vaulttokenauth" .) "true") (eq (include "check.vaultk8sauth" .) "true") -}} +{{- print true -}} +{{- end -}} +{{- end -}} + + +{{/* +Checks and enforces only 1 set of cloud creds is specified +*/}} +{{- define "enforce.singlecloudcreds" -}} +{{- $count := dict "count" (int 0) -}} +{{- $main := . -}} +{{- range $ind, $cloud_provider := include "k10.cloudProviders" . | splitList " " }} +{{ if eq (include (printf "check.%screds" $cloud_provider) $main) "true" }} +{{ $c := add1 $count.count | set $count "count" }} +{{ if gt $count.count 1 }} +{{- fail "Credentials for different cloud providers were provided but only one is allowed. Please verify your .secrets.* values." }} +{{ end }} +{{ end }} +{{- end }} +{{- end -}} + +{{/* +Converts .Values.features into k10-features: map[string]: "value" +*/}} +{{- define "k10.features" -}} +{{ range $n, $v := .Values.features }} +{{ $n }}: {{ $v | quote -}} +{{ end }} +{{- end -}} + +{{/* +Checks if string is base64 encoded +*/}} +{{- define "check.isBase64" -}} +{{- not (. | b64dec | contains "illegal base64 data") -}} +{{- end -}} + +{{/* +Returns a license base64 either from file or from values +or prints it for awsmarketplace or awsManagedLicense +*/}} +{{- define "k10.getlicense" -}} +{{- if .Values.metering.awsMarketplace -}} + {{- print "Y3VzdG9tZXJOYW1lOiBhd3MtbWFya2V0cGxhY2UKZGF0ZUVuZDogJzIxMDAtMDEtMDFUMDA6MDA6MDAuMDAwWicKZGF0ZVN0YXJ0OiAnMjAxOC0wOC0wMVQwMDowMDowMC4wMDBaJwpmZWF0dXJlczoKICBjbG91ZE1ldGVyaW5nOiBhd3MKaWQ6IGF3cy1ta3QtNWMxMDlmZDUtYWI0Yy00YTE0LWJiY2QtNTg3MGU2Yzk0MzRiCnByb2R1Y3Q6IEsxMApyZXN0cmljdGlvbnM6IG51bGwKdmVyc2lvbjogdjEuMC4wCnNpZ25hdHVyZTogY3ZEdTNTWHljaTJoSmFpazR3THMwTk9mcTNFekYxQ1pqLzRJMUZVZlBXS0JETHpuZmh2eXFFOGUvMDZxNG9PNkRoVHFSQlY3VFNJMzVkQzJ4alllaGp3cWwxNHNKT3ZyVERKZXNFWVdyMVFxZGVGVjVDd21HczhHR0VzNGNTVk5JQXVseGNTUG9oZ2x2UlRJRm0wVWpUOEtKTzlSTHVyUGxyRjlGMnpnK0RvM2UyTmVnamZ6eTVuMUZtd24xWUNlbUd4anhFaks0djB3L2lqSGlwTGQzWVBVZUh5Vm9mZHRodGV0YmhSUGJBVnVTalkrQllnRklnSW9wUlhpYnpTaEMvbCs0eTFEYzcyTDZXNWM0eUxMWFB1SVFQU3FjUWRiYnlwQ1dYYjFOT3B3aWtKMkpsR0thMldScFE4ZUFJNU9WQktqZXpuZ3FPa0lRUC91RFBtSXFBPT0K" -}} +{{- else if or ( .Values.metering.awsManagedLicense ) ( .Values.metering.licenseConfigSecretName ) -}} + {{- print "Y3VzdG9tZXJOYW1lOiBhd3MtdG90ZW0KZGF0ZUVuZDogJzIxMDAtMDEtMDFUMDA6MDA6MDAuMDAwWicKZGF0ZVN0YXJ0OiAnMjAyMS0wOS0wMVQwMDowMDowMC4wMDBaJwpmZWF0dXJlczoKICBleHRlcm5hbExpY2Vuc2U6IGF3cwogIHByb2R1Y3RTS1U6IGI4YzgyMWQ5LWJmNDAtNDE4ZC1iYTBiLTgxMjBiZjc3ZThmOQogIGtleUZpbmdlcnByaW50OiBhd3M6Mjk0NDA2ODkxMzExOkFXUy9NYXJrZXRwbGFjZTppc3N1ZXItZmluZ2VycHJpbnQKaWQ6IGF3cy1leHQtMWUxMTVlZjMtM2YyMC00MTJlLTgzODItMmE1NWUxMTc1OTFlCnByb2R1Y3Q6IEsxMApyZXN0cmljdGlvbnM6CiAgbm9kZXM6ICczJwp2ZXJzaW9uOiB2MS4wLjAKc2lnbmF0dXJlOiBkeEtLN3pPUXdzZFBOY2I1NExzV2hvUXNWeWZSVDNHVHZ0VkRuR1Vvb2VxSGlwYStTY25HTjZSNmdmdmtWdTRQNHh4RmV1TFZQU3k2VnJYeExOTE1RZmh2NFpBSHVrYmFNd3E5UXhGNkpGSmVXbTdzQmdtTUVpWVJ2SnFZVFcyMlNoakZEU1RWejY5c2JBTXNFMUd0VTdXKytITGk0dnhybjVhYkd6RkRHZW5iRE5tcXJQT3dSa3JIdTlHTFQ1WmZTNDFUL0hBMjNZZnlsTU54MGFlK2t5TGZvZXNuK3FKQzdld2NPWjh4eE94bFRJR3RuWDZ4UU5DTk5iYjhSMm5XbmljNVd0OElEc2VDR3lLMEVVRW9YL09jNFhsWVVra3FGQ0xPdVhuWDMxeFZNZ1NFQnVEWExFd3Y3K2RlSmcvb0pMaW9EVHEvWUNuM0lnem9VR2NTMGc9PQo=" -}} +{{- else -}} + {{- $license := .Values.license -}} + {{- if eq (include "check.isBase64" $license) "false" -}} + {{- $license = $license | b64enc -}} + {{- end -}} + {{- print (default (.Files.Get "license") $license) -}} +{{- end -}} +{{- end -}} + +{{/* +Returns resource usage given a pod name and container name +*/}} +{{- define "k10.resource.request" -}} +{{- $resourceDefaultList := (include "k10.serviceResources" .main | fromYaml) }} +{{- $podName := .k10_service_pod_name }} +{{- $containerName := .k10_service_container_name }} +{{- $resourceValue := "" }} +{{- if (hasKey $resourceDefaultList $podName) }} + {{- $resourceValue = index (index $resourceDefaultList $podName) $containerName }} +{{- end }} +{{- if (hasKey .main.Values.resources $podName) }} + {{- if (hasKey (index .main.Values.resources $podName) $containerName) }} + {{- $resourceValue = index (index .main.Values.resources $podName) $containerName }} + {{- end }} +{{- end }} +{{- /* If no resource usage value was provided, do not include the resources section */}} +{{- /* This allows users to set unlimited resources by providing a service key that is empty (e.g. `--set resources.=`) */}} +{{- if $resourceValue }} +resources: +{{- $resourceValue | toYaml | trim | nindent 2 }} +{{- else if eq .main.Release.Namespace "default" }} +resources: + requests: + cpu: "0.01" +{{- end }} +{{- end -}} + +{{/* +Adds priorityClassName field according to helm values. +*/}} +{{- define "k10.priorityClassName" }} +{{- $deploymentName := .k10_deployment_name }} +{{- $defaultPriorityClassName := default "" .main.Values.defaultPriorityClassName }} +{{- $priorityClassName := $defaultPriorityClassName }} + +{{- if and (hasKey .main.Values "priorityClassName") (hasKey .main.Values.priorityClassName $deploymentName) }} + {{- $priorityClassName = index .main.Values.priorityClassName $deploymentName }} +{{- end -}} + +{{- if $priorityClassName }} +priorityClassName: {{ $priorityClassName }} +{{- end }} + +{{- end }}{{/* define "k10.priorityClassName" */}} + +{{- define "kanisterToolsResources" }} +{{- if .Values.genericVolumeSnapshot.resources.requests.memory }} +KanisterToolsMemoryRequests: {{ .Values.genericVolumeSnapshot.resources.requests.memory | quote }} +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.requests.cpu }} +KanisterToolsCPURequests: {{ .Values.genericVolumeSnapshot.resources.requests.cpu | quote }} +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.limits.memory }} +KanisterToolsMemoryLimits: {{ .Values.genericVolumeSnapshot.resources.limits.memory | quote }} +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.limits.cpu }} +KanisterToolsCPULimits: {{ .Values.genericVolumeSnapshot.resources.limits.cpu | quote }} +{{- end }} +{{- end }} + +{{- define "workerPodMetricSidecarResources" }} +{{- if not (quote .Values.workerPodMetricSidecar.resources.requests.memory | empty) }} +WorkerPodMetricSidecarMemoryRequest: {{ .Values.workerPodMetricSidecar.resources.requests.memory | quote }} +{{- else if not (quote .Values.kanisterPodMetricSidecar.resources.requests.memory | empty) }} +WorkerPodMetricSidecarMemoryRequest: {{ .Values.kanisterPodMetricSidecar.resources.requests.memory | quote }} +{{- end }} +{{- if not (quote .Values.workerPodMetricSidecar.resources.requests.cpu | empty) }} +WorkerPodMetricSidecarCPURequest: {{ .Values.workerPodMetricSidecar.resources.requests.cpu | quote }} +{{- else if not (quote .Values.kanisterPodMetricSidecar.resources.requests.cpu | empty) }} +WorkerPodMetricSidecarCPURequest: {{ .Values.kanisterPodMetricSidecar.resources.requests.cpu | quote }} +{{- end }} +{{- if not (quote .Values.workerPodMetricSidecar.resources.limits.memory | empty) }} +WorkerPodMetricSidecarMemoryLimit: {{ .Values.workerPodMetricSidecar.resources.limits.memory | quote }} +{{- else if not (quote .Values.kanisterPodMetricSidecar.resources.limits.memory | empty) }} +WorkerPodMetricSidecarMemoryLimit: {{ .Values.kanisterPodMetricSidecar.resources.limits.memory | quote }} +{{- end }} +{{- if not (quote .Values.workerPodMetricSidecar.resources.limits.cpu | empty) }} +WorkerPodMetricSidecarCPULimit: {{ .Values.workerPodMetricSidecar.resources.limits.cpu | quote }} +{{- else if not (quote .Values.kanisterPodMetricSidecar.resources.limits.cpu | empty) }} +WorkerPodMetricSidecarCPULimit: {{ .Values.kanisterPodMetricSidecar.resources.limits.cpu | quote }} +{{- end }} +{{- end }} + +{{- define "workerPodResourcesCRD" }} +{{- if .Values.workerPodCRDs.resourcesRequests.maxMemory }} +workerPodMaxMemoryRequest: {{ .Values.workerPodCRDs.resourcesRequests.maxMemory | quote }} +{{- end }} +{{- if .Values.workerPodCRDs.resourcesRequests.maxCPU }} +workerPodMaxCPURequest: {{ .Values.workerPodCRDs.resourcesRequests.maxCPU | quote }} +{{- end }} +{{- if .Values.workerPodCRDs.defaultActionPodSpec }} +workerPodDefaultAPSName: {{ .Values.workerPodCRDs.defaultActionPodSpec | quote }} +{{- end }} +{{- end }} + +{{- define "get.kanisterPodCustomLabels" -}} +{{- if .Values.kanisterPodCustomLabels }} +KanisterPodCustomLabels: {{ .Values.kanisterPodCustomLabels | quote }} +{{- end }} +{{- end }} + +{{- define "get.gvsActivationToken" }} +{{- if .Values.genericStorageBackup.token }} +GVSActivationToken: {{ .Values.genericStorageBackup.token | quote }} +{{- end }} +{{- end }} + +{{- define "get.kanisterPodCustomAnnotations" -}} +{{- if .Values.kanisterPodCustomAnnotations }} +KanisterPodCustomAnnotations: {{ .Values.kanisterPodCustomAnnotations | quote }} +{{- end }} +{{- end }} + +{{/* +Lookup and return only enabled colocated services +*/}} +{{- define "get.enabledColocatedSvcList" -}} +{{- $enabledColocatedSvcList := dict }} +{{- $colocatedList := include "get.enabledColocatedServiceLookup" . | fromYaml }} +{{- range $primary, $secondaryList := $colocatedList }} + {{- $enabledSecondarySvcList := list }} + {{- range $skip, $secondary := $secondaryList }} + {{- if or (not (hasKey $.Values.optionalColocatedServices $secondary)) ((index $.Values.optionalColocatedServices $secondary).enabled) }} + {{- $enabledSecondarySvcList = append $enabledSecondarySvcList $secondary }} + {{- end }} + {{- end }} + {{- if gt (len $enabledSecondarySvcList) 0 }} + {{- $enabledColocatedSvcList = set $enabledColocatedSvcList $primary $enabledSecondarySvcList }} + {{- end }} +{{- end }} +{{- $enabledColocatedSvcList | toYaml | trim | nindent 0}} +{{- end -}} + +{{- define "get.serviceContainersInPod" -}} +{{- $podService := .k10_service_pod }} +{{- $colocatedList := include "get.enabledColocatedServices" .main | fromYaml }} +{{- $colocatedLookupByPod := include "get.enabledColocatedSvcList" .main | fromYaml }} +{{- $containerList := list $podService }} +{{- if hasKey $colocatedLookupByPod $podService }} + {{- $containerList = concat $containerList (index $colocatedLookupByPod $podService)}} +{{- end }} +{{- $containerList | join " " }} +{{- end -}} + +{{- define "get.statefulRestServicesInPod" -}} +{{- $statefulRestSvcsInPod := list }} +{{- $podService := .k10_service_pod }} +{{- $containerList := (dict "main" .main "k10_service_pod" $podService | include "get.serviceContainersInPod" | splitList " ") }} +{{- if .main.Values.global.persistence.enabled }} + {{- range $skip, $containerInPod := $containerList }} + {{- $isRestService := has $containerInPod (include "get.enabledRestServices" $.main | splitList " ") }} + {{- $isStatelessService := has $containerInPod (include "get.enabledStatelessServices" $.main | splitList " ") }} + {{- if and $isRestService (not $isStatelessService) }} + {{- $statefulRestSvcsInPod = append $statefulRestSvcsInPod $containerInPod }} + {{- end }} + {{- end }} +{{- end }} +{{- $statefulRestSvcsInPod | join " " }} +{{- end -}} + +{{- define "k10.prefixPath" -}} + {{- if .Values.route.enabled -}} + /{{ .Values.route.path | default .Release.Name | trimPrefix "/" | trimSuffix "/" }} + {{- else if .Values.ingress.create -}} + /{{ .Values.ingress.urlPath | default .Release.Name | trimPrefix "/" | trimSuffix "/" }} + {{- else -}} + /{{ .Release.Name }} + {{- end -}} +{{- end -}} + +{{/* +Check if encryption keys are specified +*/}} +{{- define "check.primaryKey" -}} +{{- if (or .Values.encryption.primaryKey.awsCmkKeyId .Values.encryption.primaryKey.vaultTransitKeyName) -}} +{{- print true -}} +{{- end -}} +{{- end -}} + +{{- define "check.validateImagePullSecrets" -}} + {{/* Validate image pull secrets if a custom Docker config is provided */}} + {{- if (or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath ) -}} + {{- if (and .Values.grafana.enabled (not .Values.global.imagePullSecret) (not .Values.grafana.image.pullSecrets)) -}} + {{ fail "A custom Docker config was provided, but Grafana is not configured to use it. Please check that global.imagePullSecret is set correctly." }} + {{- end -}} + {{- if (and .Values.prometheus.server.enabled (not .Values.global.imagePullSecret) (not .Values.prometheus.imagePullSecrets)) -}} + {{ fail "A custom Docker config was provided, but Prometheus is not configured to use it. Please check that global.imagePullSecret is set correctly." }} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- define "k10.imagePullSecrets" }} +{{- $imagePullSecrets := list .Values.global.imagePullSecret }}{{/* May be empty, but the compact below will handle that */}} +{{- if (or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath) }} + {{- $imagePullSecrets = concat $imagePullSecrets (list "k10-ecr") }} +{{- end }} +{{- $imagePullSecrets = $imagePullSecrets | compact | uniq }} + +{{- if $imagePullSecrets }} +imagePullSecrets: + {{- range $imagePullSecrets }} + {{/* Check if the name is not empty string */}} + - name: {{ . }} + {{- end }} +{{- end }} +{{- end }} + +{{/* +k10.imagePullSecretNames gets us just the secret names that are going be used +as imagePullSecrets in the k10 services. +*/}} +{{- define "k10.imagePullSecretNames" }} +{{- $pullSecretsSpec := (include "k10.imagePullSecrets" . ) | fromYaml }} +{{- if $pullSecretsSpec }} + {{- range $pullSecretsSpec.imagePullSecrets }} + {{- $secretName := . }} + {{- printf "%s " ( $secretName.name) }} + {{- end}} +{{- end}} +{{- end }} + +{{/* +Below helper template functions are referred from chart +https://github.com/prometheus-community/helm-charts/blob/main/charts/prometheus/templates/_helpers.tpl +*/}} + +{{/* +Return kubernetes version +*/}} +{{- define "k10.kubeVersion" -}} + {{- default .Capabilities.KubeVersion.Version (regexFind "v[0-9]+\\.[0-9]+\\.[0-9]+" .Capabilities.KubeVersion.Version) -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "ingress.apiVersion" -}} + {{- if and (.Capabilities.APIVersions.Has "networking.k8s.io/v1") (semverCompare ">= 1.19.x" (include "k10.kubeVersion" .)) -}} + {{- print "networking.k8s.io/v1" -}} + {{- else if .Capabilities.APIVersions.Has "extensions/v1beta1" -}} + {{- print "extensions/v1beta1" -}} + {{- else -}} + {{- print "networking.k8s.io/v1beta1" -}} + {{- end -}} +{{- end -}} + +{{/* +Is ingress part of stable APIVersion. +*/}} +{{- define "ingress.isStable" -}} + {{- eq (include "ingress.apiVersion" .) "networking.k8s.io/v1" -}} +{{- end -}} + +{{/* +Check if `ingress.defaultBackend` is properly formatted when specified. +*/}} +{{- define "check.ingress.defaultBackend" -}} + {{- if .Values.ingress.defaultBackend -}} + {{- if and .Values.ingress.defaultBackend.service.enabled .Values.ingress.defaultBackend.resource.enabled -}} + {{- fail "Both `service` and `resource` cannot be enabled in the `ingress.defaultBackend`. Provide only one." -}} + {{- end -}} + {{- if .Values.ingress.defaultBackend.service.enabled -}} + {{- if and (not .Values.ingress.defaultBackend.service.port.name) (not .Values.ingress.defaultBackend.service.port.number) -}} + {{- fail "Provide either `name` or `number` in the `ingress.defaultBackend.service.port`." -}} + {{- end -}} + {{- if and .Values.ingress.defaultBackend.service.port.name .Values.ingress.defaultBackend.service.port.number -}} + {{- fail "Both `name` and `number` cannot be specified in the `ingress.defaultBackend.service.port`. Provide only one." -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- define "check.validatePrometheusConfig" -}} + {{if and ( and .Values.global.prometheus.external.host .Values.global.prometheus.external.port) .Values.prometheus.server.enabled}} + {{ fail "Both internal and external Prometheus configs are not allowed at same time"}} + {{- end -}} +{{- end -}} + +{{/* +Defines unique ID to be assigned to all the K10 ambassador resources. +This will ensure that the K10's ambassador does not conflict with any other ambassador instances +running in the same cluster. +*/}} +{{- define "k10.ambassadorId" -}} +"kasten.io/k10" +{{- end -}} + +{{/* Check that image.values are not set. */}} +{{- define "image.values.check" -}} + {{- if not (empty .main.Values.image) }} + + {{- $registry := .main.Values.image.registry }} + {{- $repository := .main.Values.image.repository }} + {{- if or $registry $repository }} + {{- $registry = coalesce $registry "gcr.io" }} + {{- $repository = coalesce $repository "kasten-images" }} + + {{- $oldCombinedRegistry := "" }} + {{- if hasPrefix $registry $repository }} + {{- $oldCombinedRegistry = $repository }} + {{- else }} + {{- $oldCombinedRegistry = printf "%s/%s" $registry $repository }} + {{- end }} + + {{- if ne $oldCombinedRegistry .main.Values.global.image.registry }} + {{- fail "Setting image.registry and image.repository is no longer supported use global.image.registry instead" }} + {{- end }} + {{- end }} + + {{- $tag := .main.Values.image.tag }} + {{- if $tag }} + {{- if ne $tag .main.Values.global.image.tag }} + {{- fail "Setting image.tag is no longer supported use global.image.tag instead" }} + {{- end }} + {{- end }} + + {{- $pullPolicy := .main.Values.image.pullPolicy }} + {{- if $pullPolicy }} + {{- if ne $pullPolicy .main.Values.global.image.pullPolicy }} + {{- fail "Setting image.pullPolicy is no longer supported use global.image.pullPolicy instead" }} + {{- end }} + {{- end }} + + {{- end }} +{{- end -}} + +{{/* Used to verify if Ironbank is enabled */}} +{{- define "ironbank.enabled" -}} + {{- if (.Values.global.ironbank | default dict).enabled -}} + {{- print true -}} + {{- end -}} +{{- end -}} + +{{/* Get the K10 image tag. Fails if not set correctly */}} +{{- define "get.k10ImageTag" -}} + {{- $imageTag := coalesce .Values.global.image.tag (include "k10.imageTag" .) }} + {{- if not $imageTag }} + {{- fail "global.image.tag must be set because helm chart does not include a default tag." }} + {{- else }} + {{- $imageTag }} + {{- end }} +{{- end -}} + +{{- define "get.initImage" -}} + {{- (get .Values.global.images (include "init.ImageName" .)) | default (include "init.Image" .) }} +{{- end -}} + +{{- define "init.Image" -}} + {{- printf "%s:%s" (include "init.ImageRepo" .) (include "get.k10ImageTag" .) }} +{{- end -}} + +{{- define "init.ImageRepo" -}} + {{- if .Values.global.airgapped.repository }} + {{- printf "%s/%s" .Values.global.airgapped.repository (include "init.ImageName" .) }} + {{- else if .main.Values.global.azMarketPlace }} + {{- printf "%s/%s" .Values.global.azure.images.init.registry .Values.global.azure.images.init.image }} + {{- else }} + {{- printf "%s/%s" .Values.global.image.registry (include "init.ImageName" .) }} + {{- end }} +{{- end -}} + +{{- define "init.ImageName" -}} + {{- printf "init" }} +{{- end -}} + +{{- define "k10.splitImage" -}} + {{- $split_repo_tag_and_hash := .image | splitList "@" -}} + {{- $split_repo_and_tag := $split_repo_tag_and_hash | first | splitList ":" -}} + {{- $repo := $split_repo_and_tag | first -}} + + {{- /* Error if there are extra pieces we don't understand in the image */ -}} + {{- $split_repo_tag_and_hash_len := $split_repo_tag_and_hash | len -}} + {{- $split_repo_and_tag_len := $split_repo_and_tag | len -}} + {{- if or (gt $split_repo_tag_and_hash_len 2) (gt $split_repo_and_tag_len 2) -}} + {{- fail (printf "Unsupported image format: %q (%s)" .image .path) -}} + {{- end -}} + + {{- $digest := $split_repo_tag_and_hash | rest | first -}} + {{- $tag := $split_repo_and_tag | rest | first -}} + + {{- $sha := "" -}} + {{- if $digest -}} + {{- if not ($digest | hasPrefix "sha256:") -}} + {{- fail (printf "Unsupported image ...@hash type: %q (%s)" .image .path) -}} + {{- end -}} + {{- $sha = $digest | trimPrefix "sha256:" }} + {{- end -}} + + {{- /* Split out the registry if the first component of the repo contains a "." */ -}} + {{- $registry := "" }} + {{- $split_repo := $repo | splitList "/" -}} + {{- if first $split_repo | contains "." -}} + {{- $registry = first $split_repo -}} + {{- $split_repo = rest $split_repo -}} + {{- end -}} + {{- $repo = $split_repo | join "/" -}} + + {{- + (dict + "registry" $registry + "repository" $repo + "tag" ($tag | default "") + "digest" ($digest | default "") + "sha" ($sha | default "") + ) | toJson + -}} +{{- end -}} + +{{/* Fail if Ironbank is enabled and images we don't support are turned on */}} +{{- define "k10.fail.ironbankRHMarketplace" -}} + {{- if and (include "ironbank.enabled" .) (.Values.global.rhMarketPlace) -}} + {{- fail "global.ironbank.enabled and global.rhMarketPlace cannot both be enabled at the same time" -}} + {{- end -}} +{{- end -}} + +{{/* Fail if Ironbank is enabled and images we don't support are turned on */}} +{{- define "k10.fail.ironbankGrafana" -}} + {{- if (include "ironbank.enabled" .) -}} + {{- range $key, $value := .Values.grafana.sidecar -}} + {{/* + https://go.dev/doc/go1.18: the "and" used to evaluate all conditions and not terminate early + if a predicate was met, so we must have the below as their own conditional for any customers + used go version < 1.18. + */}} + {{- if kindIs "map" $value -}} + {{- if hasKey $value "enabled" -}} + {{- if $value.enabled -}} + {{- fail (printf "Ironbank deployment does not support grafana sidecar %s" $key) -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* Fail if Ironbank is enabled and images we don't support are turned on */}} +{{- define "k10.fail.ironbankPrometheus" -}} + {{- if (include "ironbank.enabled" .) -}} + {{- $prometheusDict := pick .Values.prometheus "alertmanager" "kube-state-metrics" "prometheus-node-exporter" "prometheus-pushgateway" -}} + {{- range $key, $value := $prometheusDict -}} + {{/* + https://go.dev/doc/go1.18: the "and" used to evaluate all conditions and not terminate early + if a predicate was met, so we must have the below as their own conditional for any customers + used go version < 1.18. + */}} + {{- if kindIs "map" $value -}} + {{- if hasKey $value "enabled" -}} + {{- if $value.enabled -}} + {{- fail (printf "Ironbank deployment does not support prometheus %s" $key) -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* Fail if FIPS is enabled and Grafana is turned on */}} +{{- define "k10.fail.fipsGrafana" -}} + {{- if and (.Values.fips.enabled) (.Values.grafana.enabled) -}} + {{- fail "fips.enabled and grafana.enabled cannot both be enabled at the same time" -}} + {{- end -}} +{{- end -}} + +{{/* Fail if FIPS is enabled and Prometheus is turned on */}} +{{- define "k10.fail.fipsPrometheus" -}} + {{- if and (.Values.fips.enabled) (.Values.prometheus.server.enabled) -}} + {{- fail "fips.enabled and prometheus.server.enabled cannot both be enabled at the same time" -}} + {{- end -}} +{{- end -}} + +{{/* Check to see whether SIEM logging is enabled */}} +{{- define "k10.siemEnabled" -}} + {{- if or .Values.siem.logging.cluster.enabled .Values.siem.logging.cloud.awsS3.enabled -}} + {{- true -}} + {{- end -}} +{{- end -}} + +{{/* Determine if logging should go to filepath instead of stdout */}} +{{- define "k10.siemLoggingClusterFile" -}} + {{- if .Values.siem.logging.cluster.enabled -}} + {{- if (.Values.siem.logging.cluster.file | default dict).enabled -}} + {{- .Values.siem.logging.cluster.file.path | default "" -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* Determine if a max file size should be used */}} +{{- define "k10.siemLoggingClusterFileSize" -}} + {{- if .Values.siem.logging.cluster.enabled -}} + {{- if (.Values.siem.logging.cluster.file | default dict).enabled -}} + {{- .Values.siem.logging.cluster.file.size | default "" -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* Returns a generated name for the OpenShift Service Account secret */}} +{{- define "get.openshiftServiceAccountSecretName" -}} + {{ printf "%s-k10-secret" (include "get.openshiftServiceAccountName" .) | quote }} +{{- end -}} + +{{/* +Returns a generated name for the OpenShift Service Account if a service account name +is not configuredby the user using the helm value auth.openshift.serviceAccount +*/}} +{{- define "get.openshiftServiceAccountName" -}} + {{ default (include "k10.dexServiceAccountName" .) .Values.auth.openshift.serviceAccount}} +{{- end -}} + +{{/* +Returns the required environment variables to enforce FIPS mode using +the Microsoft Go toolchain and Red Hat's OpenSSL. +*/}} +{{- define "k10.enforceFIPSEnvironmentVariables" }} +- name: GOFIPS + value: "1" +- name: OPENSSL_FORCE_FIPS_MODE + value: "1" +{{- if .Values.fips.disable_ems }} +- name: KASTEN_CRYPTO_POLICY + value: disable_ems +{{- end }} +{{- end }} + +{{/* +Returns a billing identifier label to be added to workloads for azure marketplace offer +*/}} +{{- define "k10.azMarketPlace.billingIdentifier" -}} + {{- if .Values.global.azMarketPlace -}} + azure-extensions-usage-release-identifier: {{.Release.Name}} + {{- end -}} +{{- end -}} + +{{/* +Returns the grafana URL based on the fields grafana.enabled and grafana.external.url, or in other +words based on the fact that internal grafana is used to external grafana's URL is provided +*/}} +{{- define "k10.grafanaUrl" -}} + {{- if and (.Values.grafana.enabled) (.Values.grafana.external.url) }} + {{- fail "K10's Grafana is enabled and external Grafana's URL is also provided. URL must only be provided if grafana.enabled is set to false." }} + {{- end }} + {{- if .Values.grafana.enabled }} + {{- include "k10.prefixPath" . }}/grafana/ + {{- else -}} + {{ .Values.grafana.external.url }} + {{- end }} +{{- end }} + +{{/* Fail if internal logging is enabled and fluentbit endpoint is specified, otherwise return the fluentbit endpoint even if its empty */}} +{{- define "k10.fluentbitEndpoint" -}} + {{- if and (.Values.logging.fluentbit_endpoint) (.Values.logging.internal) -}} + {{- fail "logging.fluentbit_endpoint cannot be set if logging.internal is true" -}} + {{- end -}} + {{ .Values.logging.fluentbit_endpoint }} +{{- end -}} + +{{/* +Returns the name of the K10 OpenShift Console Plugin ConfigMap +*/}} +{{- define "k10.openShiftConsolePluginConfigMapName" -}} + {{- printf "%s-config" (include "k10.openShiftConsolePluginName" .) -}} +{{- end -}} + +{{/* +Returns the name of the K10 OpenShift Console Plugin TLS certificate +*/}} +{{- define "k10.openShiftConsolePluginTLSCertName" -}} + {{- printf "%s-tls-cert" (include "k10.openShiftConsolePluginName" .) -}} +{{- end -}} + +{{/* +Returns the name of the K10 OpenShift Console Plugin Proxy +*/}} +{{- define "k10.openShiftConsolePluginProxyName" -}} + {{- printf "%s-proxy" (include "k10.openShiftConsolePluginName" .) -}} +{{- end -}} + +{{/* +Returns the name of the K10 OpenShift Console Plugin Proxy ConfigMap +*/}} +{{- define "k10.openShiftConsolePluginProxyConfigMapName" -}} + {{- printf "%s-config" (include "k10.openShiftConsolePluginProxyName" .) -}} +{{- end -}} + +{{/* +Return the name of the K10 OpenShift Console Plugin Proxy TLS certificate +*/}} +{{- define "k10.openShiftConsolePluginProxyTLSCertName" -}} + {{- printf "%s-tls-cert" (include "k10.openShiftConsolePluginProxyName" .) -}} +{{- end -}} + +{{/* +Returns true if release is being installed to the OpenShift cluster +*/}} +{{- define "k10.isOpenShift" -}} + {{- $isOpenShift := "false" -}} + + {{- if .Capabilities.APIVersions -}} + {{- if .Capabilities.APIVersions.Has "console.openshift.io/v1" -}} + {{- $isOpenShift = "true" -}} + {{- end -}} + {{- end -}} + + {{/* We consider that K10 is being installed to OpenShift if .Values.scc.create is true */}} + {{- if .Values.scc.create -}} + {{- $isOpenShift = "true" -}} + {{- end -}} + + {{- print $isOpenShift -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/_k10_container.tpl b/charts/kasten/k10/7.0.1401/templates/_k10_container.tpl new file mode 100644 index 000000000..8e2655166 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_k10_container.tpl @@ -0,0 +1,1125 @@ +{{- define "k10-containers" }} +{{- $pod := .k10_pod }} +{{- with .main }} +{{- $main_context := . }} +{{- $colocatedList := include "get.enabledColocatedServices" . | fromYaml }} +{{- $containerList := (dict "main" $main_context "k10_service_pod" $pod | include "get.serviceContainersInPod" | splitList " ") }} + containers: +{{- range $skip, $container := $containerList }} + {{- $port := default $main_context.Values.service.externalPort (index $colocatedList $container).port }} + {{- $serviceStateful := has $container (dict "main" $main_context "k10_service_pod" $pod | include "get.statefulRestServicesInPod" | splitList " ") }} + {{- dict "main" $main_context "k10_pod" $pod "k10_container" $container "externalPort" $port "stateful" $serviceStateful | include "k10-container" }} +{{- end }} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-containers" */}} + +{{- define "k10-container" }} +{{- $pod := .k10_pod }} +{{- $service := .k10_container }} +{{- $externalPort := .externalPort }} +{{- with .main }} + - name: {{ $service }}-svc + {{- dict "main" . "k10_service" $service | include "serviceImage" | indent 8 }} + imagePullPolicy: {{ .Values.global.image.pullPolicy }} +{{- if eq $service "aggregatedapis" }} + args: + - "--secure-port={{ .Values.service.aggregatedApiPort }}" + - "--cert-dir=/tmp/apiserver.local.config/certificates/" + {{- if include "k10.siemEnabled" . }} + - "--audit-policy-file=/etc/kubernetes/{{ include "k10.aggAuditPolicyFile" .}}" + {{/* SIEM cloud logging */}} + - "--enable-k10-audit-cloud-aws-s3={{ .Values.siem.logging.cloud.awsS3.enabled }}" + - "--audit-cloud-path={{ .Values.siem.logging.cloud.path }}" + {{/* SIEM cluster logging */}} + - "--enable-k10-audit-cluster={{ .Values.siem.logging.cluster.enabled }}" + - "--audit-log-file={{ include "k10.siemLoggingClusterFile" . | default (include "k10.siemAuditLogFilePath" .) }}" + - "--audit-log-file-size={{ include "k10.siemLoggingClusterFileSize" . | default (include "k10.siemAuditLogFileSize" .) }}" + {{- end }} +{{- if .Values.useNamespacedAPI }} + - "--k10-api-domain={{ template "apiDomain" . }}" +{{- end }}{{/* .Values.useNamespacedAPI */}} +{{/* +We need this explicit conversion because installation using operator hub was failing +stating that types are not same for the equality check +*/}} +{{- else if not (eq (int .Values.service.externalPort) (int $externalPort) ) }} + args: + - "--port={{ $externalPort }}" + - "--host=0.0.0.0" +{{- end }}{{/* eq $service "aggregatedapis" */}} +{{- $podName := (printf "%s-svc" $pod) }} +{{- $containerName := (printf "%s-svc" $service) }} +{{- dict "main" . "k10_service_pod_name" $podName "k10_service_container_name" $containerName | include "k10.resource.request" | indent 8}} + ports: +{{- if eq $service "aggregatedapis" }} + - containerPort: {{ .Values.service.aggregatedApiPort }} +{{- else }} + - containerPort: {{ $externalPort }} + {{- if eq $service "controllermanager" }} + - containerPort: {{ include "k10.mcExternalPort" nil }} + {{- end }} +{{- end }} +{{- if eq $service "logging" }} + - containerPort: 24224 + protocol: TCP + - containerPort: 24225 + protocol: TCP +{{- end }} + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: false + capabilities: + drop: ["ALL"] + livenessProbe: +{{- if eq $service "aggregatedapis" }} + tcpSocket: + port: {{ .Values.service.aggregatedApiPort }} + timeoutSeconds: 5 +{{- else }} + httpGet: + path: /v0/healthz + port: {{ $externalPort }} + timeoutSeconds: 1 +{{- end }} + initialDelaySeconds: 300 +{{- if ne $service "aggregatedapis" }} + readinessProbe: + httpGet: + path: /v0/healthz + port: {{ $externalPort }} + initialDelaySeconds: 3 +{{- end }} + env: +{{- if eq $service "dashboardbff" }} + - name: {{ include "k10.disabledServicesEnvVar" . }} + value: {{ include "get.disabledServices" . | quote }} +{{- end -}} +{{- if list "dashboardbff" "executor" "garbagecollector" "controllermanager" "kanister" | has $service}} +{{- if not (eq (include "check.googleproject" . ) "true") -}} + {{- fail "secrets.googleApiKey field is required when using secrets.googleProjectId" -}} +{{- end -}} +{{- $gkeSecret := default "google-secret" .Values.secrets.googleClientSecretName }} +{{- $gkeProjectId := "kasten-gke-project" }} +{{- $gkeApiKey := "/var/run/secrets/kasten.io/kasten-gke-sa.json"}} +{{- if eq (include "check.googleCredsSecret" .) "true" }} + {{- $gkeProjectId = "google-project-id" }} + {{- $gkeApiKey = "/var/run/secrets/kasten.io/google-api-key" }} +{{- end }} +{{- if eq (include "check.googleCredsOrSecret" .) "true" }} + - name: GOOGLE_APPLICATION_CREDENTIALS + value: {{ $gkeApiKey }} +{{- end }} +{{- if eq (include "check.googleCredsOrSecret" .) "true" }} + - name: projectID + valueFrom: + secretKeyRef: + name: {{ $gkeSecret }} + key: {{ $gkeProjectId }} + optional: true +{{- end }} +{{- end }} +{{- if list "dashboardbff" "executor" "garbagecollector" "controllermanager" "kanister" | has $service}} +{{- if or (eq (include "check.azuresecret" .) "true") (eq (include "check.azurecreds" .) "true" ) }} +{{- if eq (include "check.azuresecret" .) "true" }} + - name: {{ include "k10.azureClientIDEnvVar" . }} + valueFrom: + secretKeyRef: + name: {{ .Values.secrets.azureClientSecretName }} + key: azure_client_id + - name: {{ include "k10.azureTenantIDEnvVar" . }} + valueFrom: + secretKeyRef: + name: {{ .Values.secrets.azureClientSecretName }} + key: azure_tenant_id + - name: {{ include "k10.azureClientSecretEnvVar" . }} + valueFrom: + secretKeyRef: + name: {{ .Values.secrets.azureClientSecretName }} + key: azure_client_secret +{{- else }} +{{- if and (or (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureClientSecretCreds" .) "true")) (not (eq (include "check.azureFederatedIdentity" . ) "true")) }} + - name: {{ include "k10.azureClientIDEnvVar" . }} + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_client_id +{{- end }} +{{- if eq (include "check.azureClientSecretCreds" .) "true" }} + - name: {{ include "k10.azureTenantIDEnvVar" . }} + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_tenant_id + - name: {{ include "k10.azureClientSecretEnvVar" . }} + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_client_secret +{{- end }} +{{- end }} +{{- if .Values.secrets.azureResourceGroup }} + - name: AZURE_RESOURCE_GROUP + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_resource_group +{{- end }} +{{- if .Values.secrets.azureSubscriptionID }} + - name: AZURE_SUBSCRIPTION_ID + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_subscription_id +{{- end }} +{{- if .Values.secrets.azureResourceMgrEndpoint }} + - name: AZURE_RESOURCE_MANAGER_ENDPOINT + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_resource_manager_endpoint +{{- end }} +{{- if or .Values.secrets.azureADEndpoint .Values.secrets.microsoftEntraIDEndpoint }} + - name: AZURE_AD_ENDPOINT + valueFrom: + secretKeyRef: + name: azure-creds + key: entra_id_endpoint +{{- end }} +{{- if or .Values.secrets.azureADResourceID .Values.secrets.microsoftEntraIDResourceID }} + - name: AZURE_AD_RESOURCE + valueFrom: + secretKeyRef: + name: azure-creds + key: entra_id_resource_id +{{- end }} +{{- if .Values.secrets.azureCloudEnvID }} + - name: AZURE_CLOUD_ENV_ID + valueFrom: + secretKeyRef: + name: azure-creds + key: azure_cloud_env_id +{{- end }} +{{- if eq (include "check.azureMSIWithDefaultID" .) "true" }} + - name: USE_AZURE_DEFAULT_MSI + value: "{{ .Values.azure.useDefaultMSI }}" +{{- end }} +{{- if eq (include "check.azureFederatedIdentity" .) "true" }} + - name: USE_AZURE_FEDERATED_IDENTITY + value: "{{ .Values.azure.useFederatedIdentity }}" +{{- end }} +{{- end }} +{{- end }} + +{{- /* +There are 3 valid states of the secret provided by customer: +1. Only role set +2. Both aws_access_key_id and aws_secret_access_key are set +3. All of role, aws_access_key_id and aws_secret_access_key are set. +*/}} +{{- if eq (include "check.awsSecretName" .) "true" }} + {{- $customerSecret := (lookup "v1" "Secret" .Release.Namespace .Values.secrets.awsClientSecretName )}} + {{- if $customerSecret }} + {{- if and (not $customerSecret.data.role) (not $customerSecret.data.aws_access_key_id) (not $customerSecret.data.aws_secret_access_key) }} + {{ fail "Provided secret must contain at least AWS IAM Role or AWS access key ID together with AWS secret access key"}} + {{- end }} + {{- if not (or (and $customerSecret.data.aws_access_key_id $customerSecret.data.aws_secret_access_key) (and (not $customerSecret.data.aws_access_key_id) (not $customerSecret.data.aws_secret_access_key))) }} + {{ fail "Provided secret lacks aws_access_key_id or aws_secret_access_key" }} + {{- end }} + {{- end }} +{{- end }} +{{- if list "dashboardbff" "executor" "garbagecollector" "controllermanager" "metering" "kanister" | has $service}} +{{- $awsSecretName := default "aws-creds" .Values.secrets.awsClientSecretName }} + - name: AWS_ACCESS_KEY_ID + valueFrom: + secretKeyRef: + name: {{ $awsSecretName }} + key: aws_access_key_id + optional: true + - name: AWS_SECRET_ACCESS_KEY + valueFrom: + secretKeyRef: + name: {{ $awsSecretName }} + key: aws_secret_access_key + optional: true + - name: K10_AWS_IAM_ROLE + valueFrom: + secretKeyRef: + name: {{ $awsSecretName }} + key: role + optional: true +{{- end }} +{{- if list "controllermanager" "executor" "catalog" | has $service}} +{{- if eq (include "check.gwifenabled" .) "true"}} + - name: GOOGLE_WORKLOAD_IDENTITY_FEDERATION_ENABLED + value: "true" +{{- if eq (include "check.gwifidptype" .) "true"}} + - name: GOOGLE_WORKLOAD_IDENTITY_FEDERATION_IDP + value: {{ .Values.google.workloadIdentityFederation.idp.type }} +{{- end }} +{{- if eq (include "check.gwifidpaud" .) "true"}} + - name: GOOGLE_WORKLOAD_IDENTITY_FEDERATION_AUD + value: {{ .Values.google.workloadIdentityFederation.idp.aud }} +{{- end }} +{{- end }} {{/* if eq (include "check.gwifenabled" .) "true" */}} +{{- end }} {{/* list "controllermanager" "executor" "catalog" | has $service */}} +{{- if or (eq $service "crypto") (eq $service "executor") (eq $service "dashboardbff") (eq $service "repositories") }} +{{- if eq (include "check.vaultcreds" .) "true" }} + - name: VAULT_ADDR + value: {{ .Values.vault.address }} +{{- if eq (include "check.vaultk8sauth" .) "true" }} + - name: VAULT_AUTH_ROLE + value: {{ .Values.vault.role }} + - name: VAULT_K8S_SERVICE_ACCOUNT_TOKEN_PATH + value: {{ .Values.vault.serviceAccountTokenPath }} +{{- end }} +{{- if (eq (include "check.vaulttokenauth" .) "true") }} + - name: VAULT_TOKEN + valueFrom: + secretKeyRef: + name: {{.Values.vault.secretName }} + key: vault_token +{{- end }} +{{- end }} +{{- end }} +{{- if list "dashboardbff" "executor" "garbagecollector" "controllermanager" | has $service}} +{{- if or (eq (include "check.vspherecreds" .) "true") (eq (include "check.vsphereClientSecret" .) "true") }} +{{- $vsphereSecretName := default "vsphere-creds" .Values.secrets.vsphereClientSecretName }} + - name: VSPHERE_ENDPOINT + valueFrom: + secretKeyRef: + name: {{ $vsphereSecretName }} + key: vsphere_endpoint + - name: VSPHERE_USERNAME + valueFrom: + secretKeyRef: + name: {{ $vsphereSecretName }} + key: vsphere_username + - name: VSPHERE_PASSWORD + valueFrom: + secretKeyRef: + name: {{ $vsphereSecretName }} + key: vsphere_password +{{- end }} +{{- end }} + - name: VERSION + valueFrom: + configMapKeyRef: + name: k10-config + key: version + - name: {{ include "k10.fluentbitEndpointEnvVar" . }} + valueFrom: + configMapKeyRef: + name: k10-config + key: fluentbitEndpoint + optional: true +{{- if .Values.clusterName }} + - name: CLUSTER_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: clustername +{{- end }} +{{- if .Values.fips.enabled }} + {{- include "k10.enforceFIPSEnvironmentVariables" . | indent 10 }} +{{- end }} + {{- with $capabilities := include "k10.capabilities" . }} + - name: K10_CAPABILITIES + value: {{ $capabilities | quote }} + {{- end }} + {{- with $capabilities_mask := include "k10.capabilities_mask" . }} + - name: K10_CAPABILITIES_MASK + value: {{ $capabilities_mask | quote }} + {{- end }} + - name: K10_HOST_SVC + value: {{ $pod }} +{{- if eq $service "controllermanager" }} + - name: K10_STATEFUL + value: "{{ .Values.global.persistence.enabled }}" +{{- if .Values.workerPodCRDs.resourcesRequests.maxMemory }} + - name: EPHEMERAL_POD_MAX_MEMORY_REQUESTS + valueFrom: + configMapKeyRef: + name: k10-config + key: workerPodMaxMemoryRequest +{{- end }} +{{- if .Values.workerPodCRDs.resourcesRequests.maxCPU }} + - name: EPHEMERAL_POD_MAX_CPU_REQUESTS + valueFrom: + configMapKeyRef: + name: k10-config + key: workerPodMaxCPURequest +{{- end }} +{{- end }} +{{- if eq $service "executor" }} + - name: KUBEVIRT_VM_UNFREEZE_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: kubeVirtVMsUnFreezeTimeout +{{- end }} +{{- if eq $service "executor" }} + - name: QUICK_DISASTER_RECOVERY_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: quickDisasterRecoveryEnabled +{{- end }} +{{- if or (eq $service "executor") (eq $service "controllermanager") }} +{{- if or .Values.global.imagePullSecret (or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath) }} + - name: IMAGE_PULL_SECRET_NAMES + value: {{ (trimSuffix " " (include "k10.imagePullSecretNames" .)) | toJson }} +{{- end }} +{{- end }} + - name: MODEL_STORE_DIR +{{- if or (eq $service "state") (not .Values.global.persistence.enabled) }} + value: "/tmp/k10store" +{{- else }} + valueFrom: + configMapKeyRef: + name: k10-config + key: modelstoredirname +{{- end }} +{{- if or (eq $service "kanister") (eq $service "executor")}} + - name: DATA_MOVER_IMAGE + value: {{ include "get.datamoverImage" . }} +{{- end }} +{{- if or (eq $service "kanister") (eq $service "executor")}} + - name: DATA_STORE_LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: DataStoreLogLevel + - name: DATA_STORE_FILE_LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: DataStoreFileLogLevel +{{- end }} + - name: LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: loglevel +{{- if .Values.kanisterPodCustomLabels }} + - name: KANISTER_POD_CUSTOM_LABELS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodCustomLabels +{{- end }} +{{- if .Values.kanisterPodCustomAnnotations }} + - name: KANISTER_POD_CUSTOM_ANNOTATIONS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodCustomAnnotations +{{- end }} + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: K10_LIMITER_SNAPSHOT_EXPORTS_PER_ACTION + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterSnapshotExportsPerAction + - name: K10_LIMITER_WORKLOAD_SNAPSHOTS_PER_ACTION + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterWorkloadSnapshotsPerAction + - name: K10_DATA_STORE_PARALLEL_UPLOAD + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreParallelUpload + - name: K10_DATA_STORE_PARALLEL_DOWNLOAD + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreParallelDownload + - name: K10_DATA_STORE_GENERAL_CONTENT_CACHE_SIZE_MB + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreGeneralContentCacheSizeMB + - name: K10_DATA_STORE_GENERAL_METADATA_CACHE_SIZE_MB + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreGeneralMetadataCacheSizeMB + - name: K10_DATA_STORE_RESTORE_CONTENT_CACHE_SIZE_MB + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreRestoreContentCacheSizeMB + - name: K10_DATA_STORE_RESTORE_METADATA_CACHE_SIZE_MB + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreRestoreMetadataCacheSizeMB + - name: K10_LIMITER_GENERIC_VOLUME_BACKUPS_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterGenericVolumeBackupsPerCluster + - name: K10_LIMITER_SNAPSHOT_EXPORTS_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterSnapshotExportsPerCluster + - name: K10_LIMITER_VOLUME_RESTORES_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterVolumeRestoresPerCluster + - name: K10_LIMITER_CSI_SNAPSHOTS_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterCsiSnapshotsPerCluster + - name: K10_LIMITER_DIRECT_SNAPSHOTS_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterDirectSnapshotsPerCluster + - name: K10_LIMITER_IMAGE_COPIES_PER_CLUSTER + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterImageCopiesPerCluster + - name: K10_EPHEMERAL_PVC_OVERHEAD + valueFrom: + configMapKeyRef: + name: k10-config + key: K10EphemeralPVCOverhead + - name: K10_PERSISTENCE_STORAGE_CLASS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10PersistenceStorageClass + - name: AWS_ASSUME_ROLE_DURATION + valueFrom: + configMapKeyRef: + name: k10-config + key: AWSAssumeRoleDuration +{{- if (list "kanister" "executor" "repositories" | has $service) }} + - name: K10_DATA_STORE_DISABLE_COMPRESSION + valueFrom: + configMapKeyRef: + name: k10-config + key: k10DataStoreDisableCompression + + - name: K10_KANISTER_POD_METRICS_IMAGE + value: {{ include "get.metricSidecarImage" . }} + + - name: K10_TIMEOUT_WORKER_POD_READY + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutWorkerPodReady + + - name: WORKER_POD_METRIC_SIDECAR_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarEnabled + - name: WORKER_POD_METRIC_SIDECAR_METRICS_INTERVAL + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodPushgatewayMetricsInterval + {{- if or (not (quote .Values.workerPodMetricSidecar.resources.requests.memory | empty)) (not (quote .Values.kanisterPodMetricSidecar.resources.requests.memory | empty)) }} + - name: K10_WORKER_POD_METRIC_SIDECAR_MEMORY_REQUEST + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarMemoryRequest + {{- end }} + {{- if or (not (quote .Values.workerPodMetricSidecar.resources.requests.cpu | empty)) (not (quote .Values.kanisterPodMetricSidecar.resources.requests.cpu | empty)) }} + - name: K10_WORKER_POD_METRIC_SIDECAR_CPU_REQUEST + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarCPURequest + {{- end }} + {{- if or (not (quote .Values.workerPodMetricSidecar.resources.limits.memory | empty)) (not (quote .Values.kanisterPodMetricSidecar.resources.limits.memory | empty)) }} + - name: K10_WORKER_POD_METRIC_SIDECAR_MEMORY_LIMIT + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarMemoryLimit + {{- end }} + {{- if or (not (quote .Values.workerPodMetricSidecar.resources.limits.cpu | empty)) (not (quote .Values.kanisterPodMetricSidecar.resources.limits.cpu | empty)) }} + - name: K10_WORKER_POD_METRIC_SIDECAR_CPU_LIMIT + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarCPULimit + {{- end }} + {{- if .Values.scc.create }} + - name: {{ include "k10.sccNameEnvVar" . }} + value: {{ .Release.Name }}-scc + {{- end }} + {{- if .Values.workerPodCRDs.enabled }} + - name: K10_EPHEMERAL_POD_SPECS_CR_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: workerPodResourcesCRDEnabled + {{- if .Values.workerPodCRDs.defaultActionPodSpec }} + - name: K10_DEFAULT_ACTION_POD_SPEC_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: workerPodDefaultAPSName + {{- end }} + {{- end }} +{{- end }} +{{- if (list "kanister" "executor" "repositories" "crypto" "dashboardbff" "aggregatedapis" | has $service) }} + {{- if .Values.global.podLabels }} + - name: K10_CUSTOM_POD_LABELS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10CustomPodLabels + {{- end }} + {{- if .Values.global.podAnnotations }} + - name: K10_CUSTOM_POD_ANNOTATIONS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10CustomPodAnnotations + {{- end }} +{{- end }} +{{- if (list "dashboardbff" "catalog" "executor" "crypto" | has $service) }} + {{- if .Values.metering.mode }} + - name: K10REPORTMODE + value: {{ .Values.metering.mode }} + {{- end }} +{{- end }} +{{- if eq $service "garbagecollector" }} + - name: K10_GC_DAEMON_PERIOD + valueFrom: + configMapKeyRef: + name: k10-config + key: K10GCDaemonPeriod + - name: K10_GC_KEEP_MAX_ACTIONS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10GCKeepMaxActions + - name: K10_GC_ACTIONS_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: K10GCActionsEnabled +{{- end }} +{{- if (eq $service "executor") }} + - name: K10_LIMITER_EXECUTOR_THREADS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterExecutorThreads + - name: K10_LIMITER_CSI_SNAPSHOT_RESTORES_PER_ACTION + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterCsiSnapshotRestoresPerAction + - name: K10_LIMITER_VOLUME_RESTORES_PER_ACTION + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterVolumeRestoresPerAction + - name: K10_LIMITER_WORKLOAD_RESTORES_PER_ACTION + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterWorkloadRestoresPerAction + - name: K10_TIMEOUT_BLUEPRINT_BACKUP + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutBlueprintBackup + - name: K10_TIMEOUT_BLUEPRINT_RESTORE + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutBlueprintRestore + - name: K10_TIMEOUT_BLUEPRINT_DELETE + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutBlueprintDelete + - name: K10_TIMEOUT_BLUEPRINT_HOOKS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutBlueprintHooks + - name: K10_TIMEOUT_CHECK_REPO_POD_READY + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutCheckRepoPodReady + - name: K10_TIMEOUT_STATS_POD_READY + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutStatsPodReady + - name: K10_TIMEOUT_EFS_RESTORE_POD_READY + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutEFSRestorePodReady + - name: KANISTER_MANAGED_DATA_SERVICES_BLUEPRINTS_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterManagedDataServicesBlueprintsEnabled +{{- if or .Values.maxJobWaitDuration .Values.timeout.jobWait }} + - name: K10_TIMEOUT_JOB_WAIT + valueFrom: + configMapKeyRef: + name: k10-config + key: K10TimeoutJobWait +{{- end }} + - name: K10_FORCE_ROOT_IN_BLUEPRINT_ACTIONS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10ForceRootInBlueprintActions +{{- end }} +{{- if and (eq $service "executor") (.Values.awsConfig.efsBackupVaultName) }} + - name: EFS_BACKUP_VAULT_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: efsBackupVaultName +{{- end }} +{{- if and (eq $service "executor") (.Values.genericStorageBackup.token) }} + - name: K10_GVS_ACTIVATION_TOKEN + valueFrom: + configMapKeyRef: + name: k10-config + key: GVSActivationToken +{{- end }} +{{- if and (eq $service "executor") (.Values.genericStorageBackup.overridepubkey) }} + - name: OVERRIDE_GVS_TOKEN_VERIFICATION_KEY + valueFrom: + configMapKeyRef: + name: k10-config + key: overridePublicKeyForGVS +{{- end }} +{{- if and (eq $service "executor") (.Values.vmWare.taskTimeoutMin) }} + - name: VMWARE_GOM_TIMEOUT_MIN + valueFrom: + configMapKeyRef: + name: k10-config + key: vmWareTaskTimeoutMin +{{- end }} +{{- if .Values.useNamespacedAPI }} + - name: K10_API_DOMAIN + valueFrom: + configMapKeyRef: + name: k10-config + key: apiDomain +{{- end }} +{{- if .Values.jaeger.enabled }} + - name: JAEGER_AGENT_HOST + value: {{ .Values.jaeger.agentDNS }} +{{- end }} +{{- if .Values.auth.tokenAuth.enabled }} + - name: TOKEN_AUTH + valueFrom: + secretKeyRef: + name: k10-token-auth + key: auth +{{- end }} + - name: KANISTER_TOOLS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsImage +{{- with (include "k10.cacertconfigmapname" .) }} + - name: CACERT_CONFIGMAP_NAME + value: {{ . }} +{{- end }} + - name: K10_RELEASE_NAME + value: {{ .Release.Name }} + - name: KANISTER_FUNCTION_VERSION + valueFrom: + configMapKeyRef: + name: k10-config + key: kanisterFunctionVersion +{{- if and (eq $service "controllermanager") (.Values.injectKanisterSidecar.enabled) }} + - name: K10_MUTATING_WEBHOOK_ENABLED + value: "true" + - name: K10_MUTATING_WEBHOOK_TLS_CERT_DIR + valueFrom: + configMapKeyRef: + name: k10-config + key: K10MutatingWebhookTLSCertDir + - name: K10_MUTATING_WEBHOOK_PORT + value: {{ .Values.injectKanisterSidecar.webhookServer.port | quote }} +{{- end }} +{{- if (list "controllermanager" "kanister" "executor" "dashboardbff" "repositories" | has $service) }} + - name: K10_DEFAULT_PRIORITY_CLASS_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: K10DefaultPriorityClassName +{{- if .Values.genericVolumeSnapshot.resources.requests.memory }} + - name: KANISTER_TOOLS_MEMORY_REQUESTS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsMemoryRequests +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.requests.cpu }} + - name: KANISTER_TOOLS_CPU_REQUESTS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsCPURequests +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.limits.memory }} + - name: KANISTER_TOOLS_MEMORY_LIMITS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsMemoryLimits +{{- end }} +{{- if .Values.genericVolumeSnapshot.resources.limits.cpu }} + - name: KANISTER_TOOLS_CPU_LIMITS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsCPULimits +{{- end }} +{{- end }} +{{- if (list "dashboardbff" "controllermanager" "executor" | has $service) }} + {{- if .Values.prometheus.server.enabled }} + - name: K10_PROMETHEUS_HOST + value: {{ include "k10.prometheus.service.name" . }}-exp + - name: K10_PROMETHEUS_PORT + value: {{ .Values.prometheus.server.service.servicePort | quote }} + - name: K10_PROMETHEUS_BASE_URL + value: {{ .Values.prometheus.server.baseURL }} + {{- else -}} + {{- if and .Values.global.prometheus.external.host .Values.global.prometheus.external.port}} + - name: K10_PROMETHEUS_HOST + value: {{ .Values.global.prometheus.external.host }} + - name: K10_PROMETHEUS_PORT + value: {{ .Values.global.prometheus.external.port | quote }} + - name: K10_PROMETHEUS_BASE_URL + value: {{ .Values.global.prometheus.external.baseURL }} + {{- end -}} + {{- end }} + {{- if or (.Values.grafana.enabled) (.Values.grafana.external.url) }} + - name: GRAFANA_URL + value: {{ include "k10.grafanaUrl" . }} + {{- end }} +{{- end }} +{{- if eq $service "dashboardbff" }} + {{- if ne .Values.global.persistence.diskSpaceAlertPercent nil }} + - name: K10_DISK_SPACE_ALERT_PERCENT + value: {{ .Values.global.persistence.diskSpaceAlertPercent | quote }} + {{- end -}} +{{- end -}} +{{- if eq $service "controllermanager" }} + {{- if .Values.multicluster.primary.create }} + {{- if not .Values.multicluster.enabled }} + {{- fail "Cannot setup cluster as primary without enabling feature with multicluster.enabled=true" -}} + {{- end }} + {{- if not .Values.multicluster.primary.name }} + {{- fail "Cannot setup cluster as primary without setting cluster name with multicluster.primary.name" -}} + {{- end }} + {{- if not .Values.multicluster.primary.ingressURL }} + {{- fail "Cannot setup cluster as primary without providing an ingress with multicluster.primary.ingressURL" -}} + {{- end }} + - name: K10_MC_CREATE_PRIMARY + value: "true" + - name: K10_MC_PRIMARY_NAME + value: {{ .Values.multicluster.primary.name | quote }} + - name: K10_MC_PRIMARY_INGRESS_URL + value: {{ .Values.multicluster.primary.ingressURL | quote }} + {{- end }} +{{- end -}} +{{- if or $.stateful (or (eq (include "check.googleCredsOrSecret" .) "true") (eq $service "auth" "logging")) }} + volumeMounts: +{{- else if or (or (eq (include "basicauth.check" .) "true") (or .Values.auth.oidcAuth.enabled (eq (include "check.dexAuth" .) "true"))) .Values.features }} + volumeMounts: +{{- else if and (eq $service "controllermanager") (.Values.injectKanisterSidecar.enabled) }} + volumeMounts: +{{- else if or (eq (include "check.cacertconfigmap" .) "true") (include "k10.ocpcacertsautoextraction" .) }} + volumeMounts: +{{- else if eq $service "frontend" }} + volumeMounts: +{{- else if and (list "controllermanager" "executor" | has $pod) (eq (include "check.projectSAToken" .) "true")}} + volumeMounts: +{{- else if and (eq $service "aggregatedapis") (include "k10.siemEnabled" .) }} + volumeMounts: +{{- end }} +{{- if $.stateful }} + - name: {{ $service }}-persistent-storage + mountPath: {{ .Values.global.persistence.mountPath | quote }} +{{- end }} +{{- if .Values.features }} + - name: k10-features + mountPath: "/mnt/k10-features" +{{- end }} +{{- if eq $service "logging" }} + - name: logging-configmap-storage + mountPath: "/mnt/conf" +{{- end }} +{{- if and (eq $service "controllermanager") (.Values.injectKanisterSidecar.enabled) }} + - name: mutating-webhook-certs + mountPath: /etc/ssl/certs/webhook + readOnly: true +{{- end }} +{{- if list "dashboardbff" "auth" "controllermanager" | has $service}} +{{- if eq (include "basicauth.check" .) "true" }} + - name: k10-basic-auth + mountPath: "/var/run/secrets/kasten.io/k10-basic-auth" + readOnly: true +{{- end }} +{{- if (or .Values.auth.oidcAuth.enabled (eq (include "check.dexAuth" .) "true")) }} + - name: {{ include "k10.oidcSecretName" .}} + mountPath: {{ printf "%s/%s" (include "k10.secretsDir" .) (include "k10.oidcSecretName" .) }} + readOnly: true +{{- if .Values.auth.oidcAuth.clientSecretName }} + - name: {{ include "k10.oidcCustomerSecretName" .}} + mountPath: {{ printf "%s/%s" (include "k10.secretsDir" .) (include "k10.oidcCustomerSecretName" .) }} + readOnly: true +{{- end }} +{{- end }} +{{- end }} +{{- if eq (include "check.googleCredsOrSecret" .) "true"}} + - name: service-account + mountPath: {{ include "k10.secretsDir" .}} +{{- end }} +{{- if and (list "controllermanager" "executor" | has $pod) (eq (include "check.projectSAToken" .) "true")}} + - name: bound-sa-token + mountPath: "/var/run/secrets/kasten.io/serviceaccount/GWIF" + readOnly: true +{{- end }} +{{- with (include "k10.cacertconfigmapname" .) }} + - name: {{ . }} + mountPath: "/etc/ssl/certs/custom-ca-bundle.pem" + subPath: custom-ca-bundle.pem +{{- end }} +{{- if eq $service "frontend" }} + - name: frontend-config + mountPath: /etc/nginx/nginx.conf + subPath: nginx.conf + readOnly: true + - name: frontend-config + mountPath: /etc/nginx/conf.d/frontend.conf + subPath: frontend.conf + readOnly: true +{{- end}} +{{- if and (eq $service "aggregatedapis") (include "k10.siemEnabled" .) }} + - name: aggauditpolicy-config + mountPath: /etc/kubernetes/{{ include "k10.aggAuditPolicyFile" .}} + subPath: {{ include "k10.aggAuditPolicyFile" .}} + readOnly: true +{{- end}} +{{- if and (eq $service "catalog") $.stateful }} + - name: kanister-sidecar + image: {{ include "get.kanisterToolsImage" .}} + imagePullPolicy: {{ .Values.kanisterToolsImage.pullPolicy }} +{{- dict "main" . "k10_service_pod_name" $podName "k10_service_container_name" "kanister-sidecar" | include "k10.resource.request" | indent 8}} + env: + {{- with $capabilities := include "k10.capabilities" . }} + - name: K10_CAPABILITIES + value: {{ $capabilities | quote }} + {{- end }} + {{- with $capabilities_mask := include "k10.capabilities_mask" . }} + - name: K10_CAPABILITIES_MASK + value: {{ $capabilities_mask | quote }} + {{- end }} +{{- if .Values.fips.enabled }} + {{- include "k10.enforceFIPSEnvironmentVariables" . | nindent 10 }} +{{- end }} + volumeMounts: + - name: {{ $service }}-persistent-storage + mountPath: {{ .Values.global.persistence.mountPath | quote }} +{{- with (include "k10.cacertconfigmapname" .) }} + - name: {{ . }} + mountPath: "/etc/ssl/certs/custom-ca-bundle.pem" + subPath: custom-ca-bundle.pem +{{- end }} +{{- if eq (include "check.projectSAToken" .) "true" }} + - name: bound-sa-token + mountPath: "/var/run/secrets/kasten.io/serviceaccount/GWIF" + readOnly: true +{{- end }} +{{- end }} {{/* and (eq $service "catalog") $.stateful */}} +{{- if and ( eq $service "auth" ) ( eq (include "check.dexAuth" .) "true" ) }} + - name: dex + image: {{ include "get.dexImage" . }} +{{- if .Values.auth.ldap.enabled }} + command: ["/usr/local/bin/dex", "serve", "/dex-config/config.yaml"] +{{- if .Values.fips.enabled }} + env: + {{- include "k10.enforceFIPSEnvironmentVariables" . | nindent 10 }} +{{- end }} +{{- else if .Values.auth.openshift.enabled }} + {{- /* + In the case of OpenShift, a template config is used instead of a plain config for Dex. + It requires a different command to be processed correctly. + */}} + command: ["/usr/local/bin/docker-entrypoint", "dex", "serve", "/etc/dex/cfg/config.yaml"] + env: + - name: {{ include "k10.openShiftClientSecretEnvVar" . }} +{{- if and (not .Values.auth.openshift.clientSecretName) (not .Values.auth.openshift.clientSecret) }} + valueFrom: + secretKeyRef: + name: {{ include "get.openshiftServiceAccountSecretName" . }} + key: token +{{- else if .Values.auth.openshift.clientSecretName }} + valueFrom: + secretKeyRef: + name: {{ .Values.auth.openshift.clientSecretName }} + key: token +{{- else }} + value: {{ .Values.auth.openshift.clientSecret }} +{{- end }} +{{- if .Values.fips.enabled }} + {{- include "k10.enforceFIPSEnvironmentVariables" . | indent 10 }} +{{- end }} +{{- end }} + ports: + - name: http + containerPort: 8080 + volumeMounts: +{{- if .Values.auth.ldap.enabled }} + - name: dex-config + mountPath: /dex-config + - name: k10-logos-dex + mountPath: {{ include "k10.dexFrontendDir" . }}/themes/custom/ +{{- else }} + - name: config + mountPath: /etc/dex/cfg +{{- end }} +{{- with (include "k10.cacertconfigmapname" .) }} + - name: {{ . }} + mountPath: "/etc/ssl/certs/custom-ca-bundle.pem" + subPath: custom-ca-bundle.pem +{{- end }} +{{- end }} {{/* end of dex check */}} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-container" */}} + +{{- define "k10-init-container-header" }} +{{- $pod := .k10_pod }} +{{- with .main }} +{{- $main_context := . }} +{{- $containerList := (dict "main" $main_context "k10_service_pod" $pod | include "get.serviceContainersInPod" | splitList " ") }} +{{- $needsInitContainersHeader := false }} +{{- range $skip, $service := $containerList }} +{{- $serviceStateful := has $service (dict "main" $main_context "k10_service_pod" $pod | include "get.statefulRestServicesInPod" | splitList " ") }} + {{- if and ( eq $service "auth" ) $main_context.Values.auth.ldap.enabled }} + {{- $needsInitContainersHeader = true }} + {{- else if $serviceStateful }} + {{- $needsInitContainersHeader = true }} + {{- end }}{{/* initContainers header needed check */}} +{{- end }}{{/* range $skip, $service := $containerList */}} +{{- if $needsInitContainersHeader }} + initContainers: +{{- end }} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-init-container-header" */}} + +{{- define "k10-init-container" }} +{{- $pod := .k10_pod }} +{{- $podName := (printf "%s-svc" $pod) }} +{{- with .main }} +{{- $main_context := . }} +{{- $containerList := (dict "main" $main_context "k10_service_pod" $pod | include "get.serviceContainersInPod" | splitList " ") }} +{{- range $skip, $service := $containerList }} +{{- $serviceStateful := has $service (dict "main" $main_context "k10_service_pod" $pod | include "get.statefulRestServicesInPod" | splitList " ") }} +{{- if and ( eq $service "auth" ) $main_context.Values.auth.ldap.enabled }} + - name: dex-init + command: + - /dex/dexconfigmerge + args: + - --config-path=/etc/dex/cfg/config.yaml + - --secret-path=/var/run/secrets/kasten.io/bind-secret/bindPW + - --new-config-path=/dex-config/config.yaml + - --secret-field=bindPW + {{- dict "main" $main_context "k10_service" $service | include "serviceImage" | indent 8 }} + {{- dict "main" $main_context "k10_service_pod_name" $podName "k10_service_container_name" "dex-init" | include "k10.resource.request" | indent 8}} + volumeMounts: + - mountPath: /etc/dex/cfg + name: config + - mountPath: /dex-config + name: dex-config + - name: bind-secret + mountPath: "/var/run/secrets/kasten.io/bind-secret" + readOnly: true +{{- else if $serviceStateful }} + - name: upgrade-init + securityContext: + capabilities: + add: + - FOWNER + - CHOWN + runAsUser: 1000 + allowPrivilegeEscalation: false + {{- dict "main" $main_context "k10_service" "upgrade" | include "serviceImage" | indent 8 }} + imagePullPolicy: {{ $main_context.Values.global.image.pullPolicy }} + {{- dict "main" $main_context "k10_service_pod_name" $podName "k10_service_container_name" "upgrade-init" | include "k10.resource.request" | indent 8}} + env: + - name: MODEL_STORE_DIR + valueFrom: + configMapKeyRef: + name: k10-config + key: modelstoredirname + volumeMounts: + - name: {{ $service }}-persistent-storage + mountPath: {{ $main_context.Values.global.persistence.mountPath | quote }} +{{- if eq $service "catalog" }} + - name: schema-upgrade-check + {{- dict "main" $main_context "k10_service" $service | include "serviceImage" | indent 8 }} + imagePullPolicy: {{ $main_context.Values.global.image.pullPolicy }} + {{- dict "main" $main_context "k10_service_pod_name" $podName "k10_service_container_name" "schema-upgrade-check" | include "k10.resource.request" | indent 8}} + env: +{{- if $main_context.Values.clusterName }} + - name: CLUSTER_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: clustername +{{- end }} + - name: INIT_CONTAINER + value: "true" + - name: K10_RELEASE_NAME + value: {{ $main_context.Release.Name }} + - name: LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: loglevel + - name: MODEL_STORE_DIR + valueFrom: + configMapKeyRef: + name: k10-config + key: modelstoredirname + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: VERSION + valueFrom: + configMapKeyRef: + name: k10-config + key: version + volumeMounts: + - name: {{ $service }}-persistent-storage + mountPath: {{ $main_context.Values.global.persistence.mountPath | quote }} +{{- end }}{{/* eq $service "catalog" */}} +{{- end }}{{/* initContainers definitions */}} +{{- end }}{{/* range $skip, $service := $containerList */}} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-init-container" */}} diff --git a/charts/kasten/k10/7.0.1401/templates/_k10_image_tag.tpl b/charts/kasten/k10/7.0.1401/templates/_k10_image_tag.tpl new file mode 100644 index 000000000..9ac59af4f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_k10_image_tag.tpl @@ -0,0 +1 @@ +{{- define "k10.imageTag" -}}7.0.14{{- end -}} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1401/templates/_k10_metering.tpl b/charts/kasten/k10/7.0.1401/templates/_k10_metering.tpl new file mode 100644 index 000000000..6dfb6a416 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_k10_metering.tpl @@ -0,0 +1,352 @@ +{{/* Generate service spec */}} +{{/* because of https://github.com/GoogleCloudPlatform/marketplace-k8s-app-tools/issues/165 +we have to start using .Values.reportingSecret instead +of correct version .Values.metering.reportingSecret */}} +{{- define "k10-metering" }} +{{ $service := .k10_service }} +{{- $podName := (printf "%s-svc" $service) }} +{{ $main := .main }} +{{- with .main }} +{{- $servicePort := .Values.service.externalPort -}} +{{- $optionalServices := .Values.optionalColocatedServices -}} +{{- $rbac := .Values.prometheus.rbac.create -}} +{{- if $.stateful }} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + namespace: {{ .Release.Namespace }} + name: {{ $service }}-pv-claim + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ $service }} +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: {{ default .Values.global.persistence.size (index .Values.global.persistence $service "size") }} +{{- if .Values.global.persistence.storageClass }} + {{- if (eq "-" .Values.global.persistence.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.global.persistence.storageClass }}" + {{- end }} +{{- end }} +--- +{{- end }}{{/* if $.stateful */}} +{{ $service_list := include "get.enabledRestServices" . | splitList " " }} +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: {{ include "fullname" . }}-metering-config +data: + config: | +{{- if .Values.metering.reportingKey }} + identities: + - name: gcp + gcp: + encodedServiceAccountKey: {{ .Values.metering.reportingKey }} +{{- end }} + metrics: + - name: node_time + type: int + passthrough: {} + endpoints: + - name: on_disk +{{- if .Values.metering.reportingKey }} + - name: servicecontrol +{{- end }} + endpoints: + - name: on_disk + disk: +{{- if .Values.global.persistence.enabled }} + reportDir: /var/reports/ubbagent/reports +{{- else }} + reportDir: /tmp/reports/ubbagent/reports +{{- end }} + expireSeconds: 3600 +{{- if .Values.metering.reportingKey }} + - name: servicecontrol + servicecontrol: + identity: gcp + serviceName: kasten-k10.mp-kasten-public.appspot.com + consumerId: {{ .Values.metering.consumerId }} +{{- end }} + prometheusTargets: | +{{- range $service_list }} +{{- if or (not (hasKey $optionalServices .)) (index $optionalServices .).enabled }} +{{- if not (eq . "executor") }} +{{ $tmpcontx := dict "main" $main "k10service" . -}} +{{ include "k10.prometheusTargetConfig" $tmpcontx | trim | indent 4 -}} +{{- end }} +{{- end }} +{{- end }} +{{- range include "get.enabledServices" . | splitList " " }} +{{- if (or (ne . "aggregatedapis") ($rbac)) }} +{{ $tmpcontx := dict "main" $main "k10service" . -}} +{{ include "k10.prometheusTargetConfig" $tmpcontx | indent 4 -}} +{{- end }} +{{- end }} +{{- range include "get.enabledAdditionalServices" . | splitList " " }} +{{- if not (eq . "frontend") }} +{{ $tmpcontx := dict "main" $main "k10service" . -}} +{{ include "k10.prometheusTargetConfig" $tmpcontx | indent 4 -}} +{{- end }} +{{- end }} + +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ .Release.Namespace }} + name: {{ $service }}-svc + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ $service }} +spec: + replicas: {{ $.replicas }} + strategy: + type: Recreate + selector: + matchLabels: +{{ include "k10.common.matchLabels" . | indent 6 }} + component: {{ $service }} + run: {{ $service }}-svc + template: + metadata: + annotations: + {{- include "k10.deploymentPodAnnotations" . | nindent 8 }} + labels: + {{- with merge (dict "requiredLabels" (dict "component" $service "run" (printf "%s-svc" $service) )) . }} + {{- include "k10.deploymentPodLabels" . | nindent 8 }} + {{- end }} + spec: + securityContext: +{{ toYaml .Values.services.securityContext | indent 8 }} + serviceAccountName: {{ template "meteringServiceAccountName" . }} + {{- dict "main" . "k10_deployment_name" $podName | include "k10.priorityClassName" | indent 6}} + {{- include "k10.imagePullSecrets" . | indent 6 }} +{{- if $.stateful }} + initContainers: + - name: upgrade-init + securityContext: + capabilities: + add: + - FOWNER + - CHOWN + runAsUser: 1000 + allowPrivilegeEscalation: false + {{- dict "main" . "k10_service" "upgrade" | include "serviceImage" | indent 8 }} + imagePullPolicy: {{ .Values.global.image.pullPolicy }} + {{- dict "main" . "k10_service_pod_name" $podName "k10_service_container_name" "upgrade-init" | include "k10.resource.request" | indent 8}} + env: + - name: MODEL_STORE_DIR + value: /var/reports/ + volumeMounts: + - name: {{ $service }}-persistent-storage + mountPath: /var/reports/ +{{- end }} + containers: + - name: {{ $service }}-svc + {{- dict "main" . "k10_service" $service | include "serviceImage" | indent 8 }} + imagePullPolicy: {{ .Values.global.image.pullPolicy }} +{{- $containerName := (printf "%s-svc" $service) }} +{{- dict "main" . "k10_service_pod_name" $podName "k10_service_container_name" $containerName | include "k10.resource.request" | indent 8}} + ports: + - containerPort: {{ .Values.service.externalPort }} + livenessProbe: + httpGet: + path: /v0/healthz + port: {{ .Values.service.externalPort }} + initialDelaySeconds: 90 + timeoutSeconds: 1 + env: + - name: VERSION + valueFrom: + configMapKeyRef: + name: k10-config + key: version + - name: {{ include "k10.fluentbitEndpointEnvVar" . }} + valueFrom: + configMapKeyRef: + name: k10-config + key: fluentbitEndpoint + optional: true + - name: KANISTER_TOOLS + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterToolsImage +{{- if .Values.clusterName }} + - name: CLUSTER_NAME + valueFrom: + configMapKeyRef: + name: k10-config + key: clustername +{{- end }} +{{- if .Values.fips.enabled }} + {{- include "k10.enforceFIPSEnvironmentVariables" . | indent 10 }} +{{- end }} + {{- with $capabilities := include "k10.capabilities" . }} + - name: K10_CAPABILITIES + value: {{ $capabilities | quote }} + {{- end }} + {{- with $capabilities_mask := include "k10.capabilities_mask" . }} + - name: K10_CAPABILITIES_MASK + value: {{ $capabilities_mask | quote }} + {{- end }} + - name: K10_HOST_SVC + value: {{ $service }} + - name: LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: loglevel + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace +{{- if .Values.useNamespacedAPI }} + - name: K10_API_DOMAIN + valueFrom: + configMapKeyRef: + name: k10-config + key: apiDomain +{{- end }} + - name: AGENT_CONFIG_FILE + value: /var/ubbagent/config.yaml + - name: AGENT_STATE_DIR +{{- if .Values.global.persistence.enabled }} + value: "/var/reports/ubbagent" +{{- else }} + value: "/tmp/reports/ubbagent" + - name: K10SYNCSTATUSDIR + value: "/tmp/reports/k10" + - name: GRACE_PERIOD_STORE + value: /tmp/reports/clustergraceperiod + - name: NODE_USAGE_STORE + value: /tmp/reports/node_usage_history +{{- end }} +{{- if .Values.metering.awsRegion }} + - name: AWS_REGION + value: {{ .Values.metering.awsRegion }} +{{- end }} +{{- if .Values.metering.mode }} + - name: K10REPORTMODE + value: {{ .Values.metering.mode }} +{{- end }} +{{- if .Values.metering.reportCollectionPeriod }} + - name: K10_REPORT_COLLECTION_PERIOD + value: {{ .Values.metering.reportCollectionPeriod | quote }} +{{- end }} +{{- if .Values.metering.reportPushPeriod }} + - name: K10_REPORT_PUSH_PERIOD + value: {{ .Values.metering.reportPushPeriod | quote }} +{{- end }} +{{- if .Values.metering.promoID }} + - name: K10_PROMOTION_ID + value: {{ .Values.metering.promoID }} +{{- end }} + +{{- if .Values.prometheus.server.enabled }} + - name: K10_PROMETHEUS_HOST + value: {{ include "k10.prometheus.service.name" . }}-exp + - name: K10_PROMETHEUS_PORT + value: {{ .Values.prometheus.server.service.servicePort | quote }} + - name: K10_PROMETHEUS_BASE_URL + value: {{ .Values.prometheus.server.baseURL }} +{{- else -}} + {{- if and .Values.global.prometheus.external.host .Values.global.prometheus.external.port}} + - name: K10_PROMETHEUS_HOST + value: {{ .Values.global.prometheus.external.host }} + - name: K10_PROMETHEUS_PORT + value: {{ .Values.global.prometheus.external.port | quote }} + - name: K10_PROMETHEUS_BASE_URL + value: {{ .Values.global.prometheus.external.baseURL }} + {{- end -}} +{{- end }} +{{- if or .Values.workerPodMetricSidecar.enabled .Values.kanisterPodMetricSidecar.enabled }} + - name: WORKER_POD_METRIC_SIDECAR_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarEnabled + - name: WORKER_POD_METRIC_SIDECAR_METRIC_LIFETIME + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodMetricSidecarMetricLifetime + - name: WORKER_POD_METRIC_SIDECAR_METRICS_INTERVAL + valueFrom: + configMapKeyRef: + name: k10-config + key: WorkerPodPushgatewayMetricsInterval +{{- end }} +{{- if .Values.reportingSecret }} + - name: AGENT_CONSUMER_ID + valueFrom: + secretKeyRef: + name: {{ .Values.reportingSecret }} + key: consumer-id + - name: AGENT_REPORTING_KEY + valueFrom: + secretKeyRef: + name: {{ .Values.reportingSecret }} + key: reporting-key + - name: K10_RELEASE_NAME + value: {{ .Release.Name }} +{{- end }} +{{- if .Values.metering.licenseConfigSecretName }} + - name: AWS_WEB_IDENTITY_REFRESH_TOKEN_FILE + value: "/var/run/secrets/product-license/license_token" + - name: AWS_ROLE_ARN + valueFrom: + secretKeyRef: + name: {{ .Values.metering.licenseConfigSecretName }} + key: iam_role +{{- end }} + volumeMounts: + - name: meter-config + mountPath: /var/ubbagent +{{- if $.stateful }} + - name: {{ $service }}-persistent-storage + mountPath: /var/reports/ +{{- end }} +{{- if .Values.metering.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" +{{- end }} +{{- if .Values.features }} + - name: k10-features + mountPath: "/mnt/k10-features" +{{- end }} + volumes: + - name: meter-config + configMap: + name: {{ include "fullname" . }}-metering-config + items: + - key: config + path: config.yaml + - key: prometheusTargets + path: prometheusTargets.yaml +{{- if .Values.features }} + - name: k10-features + configMap: + name: k10-features +{{- end }} +{{- if $.stateful }} + - name: {{ $service }}-persistent-storage + persistentVolumeClaim: + claimName: {{ $service }}-pv-claim +{{- end }} +{{- if .Values.metering.licenseConfigSecretName }} + - name: awsmp-product-license + secret: + secretName: {{ .Values.metering.licenseConfigSecretName }} +{{- end }} +--- +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-metering" */}} diff --git a/charts/kasten/k10/7.0.1401/templates/_k10_serviceimage.tpl b/charts/kasten/k10/7.0.1401/templates/_k10_serviceimage.tpl new file mode 100644 index 000000000..9a333d92c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_k10_serviceimage.tpl @@ -0,0 +1,50 @@ +{{/* +Helper to get k10 service image +The details on how these image are being generated +is in below issue +https://kasten.atlassian.net/browse/K10-4036 +*/}} +{{- define "serviceImage" -}} +{{/* +we are maintaining the field .Values.global.images to override it when +we install the chart for red hat marketplace. If we dont +have the value specified use earlier flow, if it is, use the +value that is specified. +*/}} +{{- include "image.values.check" . -}} +{{- if not .main.Values.global.rhMarketPlace }} +{{- $serviceImage := "" -}} +{{- $tagFromDefs := "" -}} +{{- if .main.Values.global.airgapped.repository }} +{{- $serviceImage = (include "get.k10ImageTag" .main) | print .main.Values.global.airgapped.repository "/" .k10_service ":" }} +{{- else if .main.Values.global.azMarketPlace }} +{{- $az_image := (get .main.Values.global.azure.images .k10_service) }} +{{- $serviceImage = print $az_image.registry "/" $az_image.image ":" $az_image.tag }} +{{- else }} +{{- $serviceImage = (include "get.k10ImageTag" .main) | print .main.Values.global.image.registry "/" .k10_service ":" }} +{{- end }}{{/* if .main.Values.global.airgapped.repository */}} +{{- $serviceImageKey := print (replace "-" "" .k10_service) "Image" }} +{{- if eq $serviceImageKey "dexImage" }} +{{- $tagFromDefs = (include "dex.dexImageTag" .) }} +{{- end }}{{/* if eq $serviceImageKey "dexImage" */}} +{{- if index .main.Values $serviceImageKey }} +{{- $service_values := index .main.Values $serviceImageKey }} +{{- if .main.Values.global.airgapped.repository }} +{{ $valuesImage := (splitList "/" (index $service_values "image")) }} +{{- if $tagFromDefs }} +image: {{ printf "%s/%s:k10-%s" .main.Values.global.airgapped.repository (index $valuesImage (sub (len $valuesImage) 1) ) $tagFromDefs -}} +{{- end }} +{{- else }}{{/* .main.Values.global.airgapped.repository */}} +{{- if $tagFromDefs }} +image: {{ printf "%s:%s" (index $service_values "image") $tagFromDefs }} +{{- else }} +image: {{ index $service_values "image" }} +{{- end }} +{{- end }}{{/* .main.Values.global.airgapped.repository */}} +{{- else }} +image: {{ $serviceImage }} +{{- end -}}{{/* index .main.Values $serviceImageKey */}} +{{- else }} +image: {{ printf "%s" (get .main.Values.global.images .k10_service) }} +{{- end }}{{/* if not .main.Values.images.executor */}} +{{- end -}}{{/* define "serviceImage" */}} diff --git a/charts/kasten/k10/7.0.1401/templates/_k10_template.tpl b/charts/kasten/k10/7.0.1401/templates/_k10_template.tpl new file mode 100644 index 000000000..48ab4d78d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_k10_template.tpl @@ -0,0 +1,245 @@ +{{/* Generate service spec */}} +{{- define "k10-default" }} +{{- $service := .k10_service }} +{{- $deploymentName := (printf "%s-svc" $service) }} +{{- with .main }} +{{- $main_context := . }} +{{- range $skip, $statefulContainer := compact (dict "main" $main_context "k10_service_pod" $service | include "get.statefulRestServicesInPod" | splitList " ") }} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + namespace: {{ $main_context.Release.Namespace }} + name: {{ $statefulContainer }}-pv-claim + labels: +{{ include "helm.labels" $main_context | indent 4 }} + component: {{ $statefulContainer }} +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: {{ index (index $main_context.Values.global.persistence $statefulContainer | default dict) "size" | default $main_context.Values.global.persistence.size }} +{{- if $main_context.Values.global.persistence.storageClass }} + {{- if (eq "-" $main_context.Values.global.persistence.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ $main_context.Values.global.persistence.storageClass }}" + {{- end }} +{{- end }} +--- +{{- end }}{{/* if $.stateful */}} +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ .Release.Namespace }} + name: {{ $deploymentName }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ $service }} +spec: + replicas: {{ $.replicas }} + strategy: + type: Recreate + selector: + matchLabels: +{{ include "k10.common.matchLabels" . | indent 6 }} + component: {{ $service }} + run: {{ $deploymentName }} + template: + metadata: + annotations: +{{- +$requiredAnnotations := (dict + "run" $deploymentName + "checksum/frontend-nginx-config" (include (print .Template.BasePath "/frontend-nginx-configmap.yaml") . | sha256sum) +) +-}} +{{- if .Values.auth.ldap.restartPod -}} + {{- $_ := set $requiredAnnotations "rollme" (randAlphaNum 5) -}} +{{- end -}} + {{- with merge (dict "requiredAnnotations" $requiredAnnotations) . }} + {{- include "k10.deploymentPodAnnotations" . | nindent 8 }} + {{- end }} + labels: +{{- +$requiredLabels := (dict + "component" $service + "run" $deploymentName +) +-}} +{{- if list "executor" "controllermanager" | has $service}} + {{- if eq (include "check.azureFederatedIdentity" .) "true" }} + azure.workload.identity/use: "true" + {{- end }} +{{- end }} + {{- with merge (dict "requiredLabels" $requiredLabels) . }} + {{- include "k10.deploymentPodLabels" . | nindent 8 }} + {{- end }} + spec: +{{- if eq $service "executor" }} +{{- if .Values.services.executor.hostNetwork }} + hostNetwork: true +{{- end }}{{/* .Values.services.executor.hostNetwork */}} +{{- end }}{{/* eq $service "executor" */}} +{{- if eq $service "aggregatedapis" }} +{{- if .Values.services.aggregatedapis.hostNetwork }} + hostNetwork: true +{{- end }}{{/* .Values.services.aggregatedapis.hostNetwork */}} +{{- end }}{{/* eq $service "aggregatedapis" */}} +{{- if eq $service "dashboardbff" }} +{{- if .Values.services.dashboardbff.hostNetwork }} + hostNetwork: true +{{- end }}{{/* .Values.services.dashboardbff.hostNetwork */}} +{{- end }}{{/* eq $service "dashboardbff" */}} + securityContext: +{{ toYaml .Values.services.securityContext | indent 8 }} + serviceAccountName: {{ template "serviceAccountName" . }} + {{- dict "main" . "k10_deployment_name" $deploymentName | include "k10.priorityClassName" | indent 6}} + {{- include "k10.imagePullSecrets" . | indent 6 }} +{{- /* initContainers: */}} +{{- (dict "main" . "k10_pod" $service | include "k10-init-container-header") }} +{{- (dict "main" . "k10_pod" $service | include "k10-init-container") }} +{{- /* containers: */}} +{{- (dict "main" . "k10_pod" $service | include "k10-containers") }} +{{- /* volumes: */}} +{{- (dict "main" . "k10_pod" $service | include "k10-deployment-volumes-header") }} +{{- (dict "main" . "k10_pod" $service | include "k10-deployment-volumes") }} +--- +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-default" */}} + +{{- define "k10-deployment-volumes-header" }} +{{- $pod := .k10_pod }} +{{- with .main }} +{{- $main_context := . }} +{{- $containerList := (dict "main" $main_context "k10_service_pod" $pod | include "get.serviceContainersInPod" | splitList " ") }} +{{- $needsVolumesHeader := false }} +{{- range $skip, $service := $containerList }} + {{- $serviceStateful := has $service (dict "main" $main_context "k10_service_pod" $pod | include "get.statefulRestServicesInPod" | splitList " ") }} + {{- if or $serviceStateful (or (eq (include "check.googlecreds" $main_context) "true") (eq $service "auth" "logging")) }} + {{- $needsVolumesHeader = true }} + {{- else if or (or (eq (include "basicauth.check" $main_context) "true") (or $main_context.Values.auth.oidcAuth.enabled (eq (include "check.dexAuth" $main_context) "true"))) $main_context.Values.features }} + {{- $needsVolumesHeader = true }} + {{- else if and (eq $service "controllermanager") ($main_context.Values.injectKanisterSidecar.enabled) }} + {{- $needsVolumesHeader = true }} + {{- else if or (eq (include "check.cacertconfigmap" $main_context) "true") (include "k10.ocpcacertsautoextraction" $main_context) }} + {{- $needsVolumesHeader = true }} + {{- else if and ( eq $service "auth" ) ( eq (include "check.dexAuth" $main_context) "true" ) }} + {{- $needsVolumesHeader = true }} + {{- else if eq $service "frontend" }} + {{- $needsVolumesHeader = true }} + {{- else if and (list "controllermanager" "executor" "catalog" | has $pod) (eq (include "check.projectSAToken" $main_context) "true")}} + {{- $needsVolumesHeader = true }} + {{- else if and (eq $service "aggregatedapis") (include "k10.siemEnabled" $main_context) }} + {{- $needsVolumesHeader = true }} + {{- end }}{{/* volumes header needed check */}} +{{- end }}{{/* range $skip, $service := $containerList */}} +{{- if $needsVolumesHeader }} + volumes: +{{- end }} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-init-container-header" */}} + +{{- define "k10-deployment-volumes" }} +{{- $pod := .k10_pod }} +{{- with .main }} +{{- if .Values.features }} + - name: k10-features + configMap: + name: k10-features +{{- end }} +{{- if list "dashboardbff" "auth" "controllermanager" | has $pod}} +{{- if eq (include "basicauth.check" .) "true" }} + - name: k10-basic-auth + secret: + secretName: {{ default "k10-basic-auth" .Values.auth.basicAuth.secretName }} +{{- end }} +{{- if .Values.auth.oidcAuth.enabled }} + - name: {{ include "k10.oidcSecretName" .}} + secret: + secretName: {{ default (include "k10.oidcSecretName" .) .Values.auth.oidcAuth.secretName }} +{{- if .Values.auth.oidcAuth.clientSecretName }} + - name: {{ include "k10.oidcCustomerSecretName" . }} + secret: + secretName: {{ .Values.auth.oidcAuth.clientSecretName }} +{{- end }} +{{- end }} +{{- if .Values.auth.openshift.enabled }} + - name: {{ include "k10.oidcSecretName" .}} + secret: + secretName: {{ default (include "k10.oidcSecretName" .) .Values.auth.openshift.secretName }} +{{- end }} +{{- if .Values.auth.ldap.enabled }} + - name: {{ include "k10.oidcSecretName" .}} + secret: + secretName: {{ default (include "k10.oidcSecretName" .) .Values.auth.ldap.secretName }} + - name: k10-logos-dex + configMap: + name: k10-logos-dex +{{- end }} +{{- end }} +{{- range $skip, $statefulContainer := compact (dict "main" . "k10_service_pod" $pod | include "get.statefulRestServicesInPod" | splitList " ") }} + - name: {{ $statefulContainer }}-persistent-storage + persistentVolumeClaim: + claimName: {{ $statefulContainer }}-pv-claim +{{- end }} +{{- if eq (include "check.googleCredsOrSecret" .) "true" }} +{{- $gkeSecret := default "google-secret" .Values.secrets.googleClientSecretName }} + - name: service-account + secret: + secretName: {{ $gkeSecret }} +{{- end }} +{{- if and (list "controllermanager" "executor" "catalog" | has $pod) (eq (include "check.projectSAToken" .) "true")}} + - name: bound-sa-token + projected: + sources: + - serviceAccountToken: +{{- if eq (include "check.gwifidpaud" .) "true" }} + audience: {{ .Values.google.workloadIdentityFederation.idp.aud }} +{{- end }} + expirationSeconds: 3600 + path: token +{{- end }} +{{- with (include "k10.cacertconfigmapname" .) }} + - name: {{ . }} + configMap: + name: {{ . }} +{{- end }} +{{- if eq $pod "frontend" }} + - name: frontend-config + configMap: + name: frontend-config +{{- end }} +{{- if and (eq $pod "aggregatedapis") (include "k10.siemEnabled" .) }} + - name: aggauditpolicy-config + configMap: + name: aggauditpolicy-config +{{- end }} +{{- $containersInThisPod := (dict "main" . "k10_service_pod" $pod | include "get.serviceContainersInPod" | splitList " ") }} +{{- if has "logging" $containersInThisPod }} + - name: logging-configmap-storage + configMap: + name: fluentbit-configmap +{{- end }} +{{- if and (has "controllermanager" $containersInThisPod) (.Values.injectKanisterSidecar.enabled) }} + - name: mutating-webhook-certs + secret: + secretName: controllermanager-certs +{{- end }} +{{- if and ( has "auth" $containersInThisPod) ( eq (include "check.dexAuth" .) "true" ) }} + - name: config + configMap: + name: k10-dex + items: + - key: config.yaml + path: config.yaml +{{- if .Values.auth.ldap.enabled }} + - name: dex-config + emptyDir: {} + - name: bind-secret + secret: + secretName: {{ default "k10-dex" .Values.auth.ldap.bindPWSecretName }} +{{- end }} +{{- end }} +{{- end }}{{/* with .main */}} +{{- end }}{{/* define "k10-init-container-header" */}} diff --git a/charts/kasten/k10/7.0.1401/templates/_prometheus.tpl b/charts/kasten/k10/7.0.1401/templates/_prometheus.tpl new file mode 100644 index 000000000..a49a8363f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/_prometheus.tpl @@ -0,0 +1,29 @@ +{{/*** MATCH LABELS *** + NOTE: The match labels here (`app` and `release`) are divergent from + the match labels set by the upstream chart. This is intentional since a + Deployment's `spec.selector` is immutable and K10 has already been shipped + with these values. + + A change to these selector labels will mean that all customers must manually + delete the Prometheus Deployment before upgrading, which is a situation we don't + want for our customers. + + Instead, the `app.kubernetes.io/name` and `app.kubernetes.io/instance` labels + are included in the `prometheus.commonMetaLabels` in: + `helm/k10/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl`. +*/}} +{{- define "prometheus.common.matchLabels" -}} +app: {{ include "prometheus.name" . }} +release: {{ .Release.Name }} +{{- end -}} + +{{- define "prometheus.server.labels" -}} +{{ include "prometheus.server.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +app.kubernetes.io/component: {{ .Values.server.name }} +{{- end -}} + +{{- define "prometheus.server.matchLabels" -}} +component: {{ .Values.server.name | quote }} +{{ include "prometheus.common.matchLabels" . }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/aggregatedaudit-policy.yaml b/charts/kasten/k10/7.0.1401/templates/aggregatedaudit-policy.yaml new file mode 100644 index 000000000..ef7f03c6c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/aggregatedaudit-policy.yaml @@ -0,0 +1,34 @@ +{{- if include "k10.siemEnabled" . -}} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: aggauditpolicy-config + namespace: {{ .Release.Namespace }} +data: + {{ include "k10.aggAuditPolicyFile" .}}: | + apiVersion: audit.k8s.io/v1 + kind: Policy + omitStages: + - "RequestReceived" + rules: + - level: RequestResponse + resources: + - group: "actions.kio.kasten.io" + resources: ["backupactions", "cancelactions", "exportactions", "importactions", "restoreactions", "retireactions", "runactions"] + - group: "apps.kio.kasten.io" + resources: ["applications", "clusterrestorepoints", "restorepoints", "restorepointcontents"] + - group: "repositories.kio.kasten.io" + resources: ["storagerepositories"] + - group: "vault.kio.kasten.io" + resources: ["passkeys"] + verbs: ["create", "update", "patch", "delete", "get"] + - level: None + nonResourceURLs: + - /healthz* + - /version + - /openapi/v2* + - /openapi/v3* + - /timeout* +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/api-tls-secrets.yaml b/charts/kasten/k10/7.0.1401/templates/api-tls-secrets.yaml new file mode 100644 index 000000000..386d3b999 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/api-tls-secrets.yaml @@ -0,0 +1,13 @@ +{{- if and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey }} +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: gateway-certs +type: kubernetes.io/tls +data: + tls.crt: {{ .Values.secrets.apiTlsCrt }} + tls.key: {{ .Values.secrets.apiTlsKey }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/apiservice.yaml b/charts/kasten/k10/7.0.1401/templates/apiservice.yaml new file mode 100644 index 000000000..1811df48a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/apiservice.yaml @@ -0,0 +1,25 @@ +{{/* Template to generate the aggregated APIService/Service objects */}} +{{- if .Values.apiservices.deployed -}} +{{- $main := . -}} +{{- $container_port := .Values.service.internalPort -}} +{{- $namespace := .Release.Namespace -}} +{{- range include "k10.aggregatedAPIs" . | splitList " " -}} +--- +apiVersion: apiregistration.k8s.io/v1 +kind: APIService +metadata: + name: v1alpha1.{{ . }}.{{ template "apiDomain" $main }} + labels: + apiserver: "true" +{{ include "helm.labels" $ | indent 4 }} +spec: + version: v1alpha1 + group: {{ . }}.{{ template "apiDomain" $main }} + groupPriorityMinimum: 2000 + service: + namespace: {{$namespace}} + name: aggregatedapis-svc + versionPriority: 10 + insecureSkipTLSVerify: true +{{ end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/daemonsets.yaml b/charts/kasten/k10/7.0.1401/templates/daemonsets.yaml new file mode 100644 index 000000000..b8c50b505 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/daemonsets.yaml @@ -0,0 +1,26 @@ +{{- if .Values.metering.redhatMarketplacePayg }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + namespace: {{ .Release.Namespace }} + name: k10-rhmp-paygo + labels: +{{ include "helm.labels" . | indent 4 }} + component: paygo +spec: + selector: + matchLabels: +{{ include "k10.common.matchLabels" . | indent 6 }} + component: paygo + template: + metadata: + labels: +{{ include "helm.labels" . | indent 8 }} + component: paygo + spec: + containers: + - name: paygo + image: {{ .Values.global.images.paygo_daemonset }} + command: [ "sleep" ] + args: [ "36500d" ] +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/deployments.yaml b/charts/kasten/k10/7.0.1401/templates/deployments.yaml new file mode 100644 index 000000000..29713d214 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/deployments.yaml @@ -0,0 +1,39 @@ +{{/* +Generates deployment specs for K10 services and other services such as +"frontend" and "kanister". +*/}} +{{- include "singleAuth.check" . -}} +{{- $main_context := . -}} +{{- $stateless_services := include "get.enabledStatelessServices" . | splitList " " -}} +{{- $colocated_services := include "get.enabledColocatedServices" . | fromYaml -}} +{{ $service_list := include "get.enabledRestServices" . | splitList " " }} +{{- range $skip, $k10_service := $service_list }} + {{ if not (hasKey $colocated_services $k10_service ) }} + {{/* Set $stateful for stateful services when .Values.global.persistence.enabled is true */}} + {{- $stateful := and $.Values.global.persistence.enabled (not (has $k10_service $stateless_services)) -}} + {{/* Use `limiter.executorReplicas` (with back-compatibility with already deprecated + `executorReplicas`) Helm parameter to configure number of replicas for service. In case of missing + `limiter.{servicename}Replicas` Helm parameter will be set `1`. + See also function `replicasFieldForService` in go/src/kasten.io/k10/kio/tools/restorectl/servicescaler/config.go.*/}} + {{- $replicas := get $.Values (printf "%sReplicas" $k10_service) | default -1 -}} + {{- $replicasInt := int $replicas -}} + {{- if le $replicasInt 0 -}} + {{- $replicas = get $.Values (printf "limiter.%sReplicas" $k10_service) | default 1 -}} + {{- end }} + {{ $tmp_contx := dict "main" $main_context "k10_service" $k10_service "stateful" $stateful "replicas" $replicas }} + {{ if eq $k10_service "metering" }} + {{- include "k10-metering" $tmp_contx -}} + {{ else }} + {{- include "k10-default" $tmp_contx -}} + {{ end }} + {{ end }}{{/* if not (hasKey $colocated_services $k10_service ) */}} +{{- end }} +{{/* +Generate deployment specs for additional services. These are stateless and have +1 replica. +*/}} +{{- range $skip, $k10_service := concat (include "get.enabledServices" . | splitList " ") (include "get.enabledAdditionalServices" . | splitList " ") }} + {{- if eq $k10_service "gateway" -}}{{- continue -}}{{- end -}} + {{ $tmp_contx := dict "main" $main_context "k10_service" $k10_service "stateful" false "replicas" 1 }} + {{- include "k10-default" $tmp_contx -}} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/fluentbit-configmap.yaml b/charts/kasten/k10/7.0.1401/templates/fluentbit-configmap.yaml new file mode 100644 index 000000000..71cecb966 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/fluentbit-configmap.yaml @@ -0,0 +1,34 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: fluentbit-configmap +data: + fluentbit.conf: | + [SERVICE] + HTTP_Server On + HTTP_Listen 0.0.0.0 + HTTP_PORT 24225 + + [INPUT] + Name tcp + Listen 0.0.0.0 + Port 24224 + + [OUTPUT] + Name stdout + Match * + + [OUTPUT] + Name file + Match * + File {{ .Values.global.persistence.mountPath }}/k10.log + logrotate.conf: | + {{ .Values.global.persistence.mountPath }}/k10.log { + create + missingok + rotate 6 + size 1G + } diff --git a/charts/kasten/k10/7.0.1401/templates/frontend-nginx-configmap.yaml b/charts/kasten/k10/7.0.1401/templates/frontend-nginx-configmap.yaml new file mode 100644 index 000000000..93d17b3a1 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/frontend-nginx-configmap.yaml @@ -0,0 +1,50 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: frontend-config +data: + frontend.conf: | + server { + listen {{ .Values.service.externalPort }} default_server; + {{- if .Values.global.network.enable_ipv6 }} + listen [::]:{{ .Values.service.externalPort }} default_server; + {{- end }} + server_name localhost; + + gzip on; + # serves *.gz files (when present) instead of dynamic compression + gzip_static on; + + root /frontend; + index index.html; + + location / { + try_files $uri $uri/ =404; + } + } + nginx.conf: | + #user nginx; # this directive is ignored if we use a non-root user in Dockerfile + worker_processes 4; + + error_log stderr warn; + pid /var/run/nginx/nginx.pid; + + events { + worker_connections 1024; + } + + http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + access_log /dev/stdout; + sendfile on; + keepalive_timeout 650; + + # turn off nginx version in responses + server_tokens off; + + include /etc/nginx/conf.d/*.conf; + } diff --git a/charts/kasten/k10/7.0.1401/templates/gateway-ext.yaml b/charts/kasten/k10/7.0.1401/templates/gateway-ext.yaml new file mode 100644 index 000000000..00da4c27b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/gateway-ext.yaml @@ -0,0 +1,36 @@ +{{/* Externally exposed service for gateway endpoint. */}} +{{- $container_port := .Values.service.internalPort -}} +{{- if .Values.externalGateway.create -}} +{{- include "authEnabled.check" . -}} +apiVersion: v1 +kind: Service +metadata: + namespace: {{ $.Release.Namespace }} + name: gateway-ext + labels: + service: gateway + {{- if eq "route53-mapper" (default " " .Values.externalGateway.fqdn.type) }} + dns: route53 + {{- end }} +{{ include "helm.labels" . | indent 4 }} + annotations: + {{- if .Values.externalGateway.annotations }} +{{ toYaml .Values.externalGateway.annotations | indent 4 }} + {{- end }} +{{ include "dnsAnnotations" . | indent 4 }} + {{- if .Values.externalGateway.awsSSLCertARN }} + service.beta.kubernetes.io/aws-load-balancer-ssl-cert: {{ .Values.externalGateway.awsSSLCertARN }} + service.beta.kubernetes.io/aws-load-balancer-ssl-ports: https + {{- if .Values.externalGateway.awsSecurityGroup }} + service.beta.kubernetes.io/aws-load-balancer-extra-security-groups: {{ .Values.externalGateway.awsSecurityGroup }} + {{- end }} + {{- end }} +spec: + type: LoadBalancer + ports: + - name: https + port: {{ if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) .Values.externalGateway.awsSSLCertARN }}443{{ else }}80{{ end }} + targetPort: {{ $container_port }} + selector: + service: gateway +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/gateway.yaml b/charts/kasten/k10/7.0.1401/templates/gateway.yaml new file mode 100644 index 000000000..23a7aa437 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/gateway.yaml @@ -0,0 +1,249 @@ +{{- $container_port := .Values.gateway.service.internalPort | default 8000 -}} +{{- $service_port := .Values.gateway.service.externalPort -}} +{{- $admin_port := default 8877 .Values.service.gatewayAdminPort -}} +--- +apiVersion: v1 +kind: Service +metadata: + namespace: {{ $.Release.Namespace }} + labels: + service: gateway +{{ include "helm.labels" . | indent 4 }} + name: gateway + {{- if not (include "k10.capability.gateway" $) }} + annotations: + getambassador.io/config: | + --- + apiVersion: getambassador.io/v3alpha1 + kind: AuthService + name: authentication + auth_service: "auth-svc:8000" + path_prefix: "/v0/authz" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + allowed_authorization_headers: + - x-cluster-name + allowed_request_headers: + - "x-forwarded-access-token" + --- + apiVersion: getambassador.io/v3alpha1 + kind: Host + name: ambassadorhost + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + {{- if .Values.secrets.tlsSecret }} + tlsSecret: + name: {{ .Values.secrets.tlsSecret }} + {{- else if and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey }} + tlsSecret: + name: gateway-certs + {{- end }} + requestPolicy: + insecure: + action: Route + --- + apiVersion: getambassador.io/v3alpha1 + kind: Listener + name: ambassadorlistener + port: {{ $container_port }} + securityModel: XFP + protocol: HTTPS + hostBinding: + namespace: + from: SELF + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + --- + {{- if (eq "endpoint" .Values.apigateway.serviceResolver) }} + apiVersion: getambassador.io/v3alpha1 + kind: KubernetesEndpointResolver + name: endpoint + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + --- + {{- end }} + apiVersion: getambassador.io/v3alpha1 + kind: Module + name: ambassador + config: + diagnostics: + enabled: false + service_port: {{ $container_port }} + {{- if .Values.global.network.enable_ipv6 }} + enable_ipv6: true + {{- end }} + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + {{- if (eq "endpoint" .Values.apigateway.serviceResolver) }} + resolver: endpoint + load_balancer: + policy: round_robin + {{- end }} + {{- end }} +spec: + ports: + - name: http + port: {{ $service_port }} + targetPort: {{ $container_port }} + selector: + service: gateway +--- +{{- if not (include "k10.capability.gateway" $) }} +{{- if .Values.gateway.exposeAdminPort }} +apiVersion: v1 +kind: Service +metadata: + namespace: {{ $.Release.Namespace }} + name: gateway-admin + labels: + service: gateway + annotations: + getambassador.io/config: | + apiVersion: getambassador.io/v3alpha1 + kind: Module + name: ambassador + config: + diagnostics: + enabled: false +{{ include "helm.labels" . | indent 4 }} +spec: + ports: + - name: metrics + port: {{ $admin_port }} + targetPort: {{ $admin_port }} + selector: + service: gateway +--- +{{- end }} +{{- end }} +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ $.Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: gateway + name: gateway +spec: + replicas: 1 + selector: + matchLabels: + service: gateway + template: + metadata: + annotations: + {{- include "k10.deploymentPodAnnotations" . | nindent 8 }} + labels: + {{- with merge (dict "requiredLabels" (dict "component" "gateway" "service" "gateway")) . }} + {{- include "k10.deploymentPodLabels" . | nindent 8 }} + {{- end }} +{{- if (include "k10.capability.gateway" $) }} + spec: + serviceAccountName: {{ template "serviceAccountName" . }} + {{- dict "main" . "k10_deployment_name" "gateway" | include "k10.priorityClassName" | indent 6}} + {{- include "k10.imagePullSecrets" . | indent 6 }} + containers: + - name: gateway + {{- dict "main" . "k10_service" "gateway" | include "serviceImage" | indent 8 }} + {{- if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) }} + volumeMounts: + - name: tls-volume + mountPath: "/etc/tls" + readOnly: true + {{- end }} + resources: + limits: + cpu: {{ .Values.gateway.resources.limits.cpu | quote }} + memory: {{ .Values.gateway.resources.limits.memory | quote }} + requests: + cpu: {{ .Values.gateway.resources.requests.cpu | quote }} + memory: {{ .Values.gateway.resources.requests.memory | quote }} + env: + - name: LOG_LEVEL + valueFrom: + configMapKeyRef: + name: k10-config + key: loglevel + - name: VERSION + valueFrom: + configMapKeyRef: + name: k10-config + key: version +{{- if .Values.fips.enabled }} + {{- include "k10.enforceFIPSEnvironmentVariables" . | indent 10 }} +{{- end }} + {{- with $capabilities := include "k10.capabilities" . }} + - name: K10_CAPABILITIES + value: {{ $capabilities | quote }} + {{- end }} + {{- with $capabilities_mask := include "k10.capabilities_mask" . }} + - name: K10_CAPABILITIES_MASK + value: {{ $capabilities_mask | quote }} + {{- end }} + {{- if eq (include "check.dexAuth" .) "true" }} + - name: {{ include "k10.gatewayEnableDex" . }} + value: "true" + {{- end }} + envFrom: + - configMapRef: + name: k10-gateway + livenessProbe: + httpGet: + path: /healthz + port: {{ $container_port }} + initialDelaySeconds: 5 + readinessProbe: + httpGet: + path: /healthz + port: {{ $container_port }} + restartPolicy: Always + {{- if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) }} + volumes: + - name: tls-volume + secret: + secretName: {{ .Values.secrets.tlsSecret | default "gateway-certs" }} + {{- end }} +{{- else }} + spec: + serviceAccountName: {{ template "serviceAccountName" . }} + {{- dict "main" . "k10_deployment_name" "gateway" | include "k10.priorityClassName" | indent 6}} + {{- include "k10.imagePullSecrets" . | indent 6 }} + containers: + - name: ambassador + image: {{ include "get.emissaryImage" . }} + resources: + limits: + cpu: {{ .Values.gateway.resources.limits.cpu | quote }} + memory: {{ .Values.gateway.resources.limits.memory | quote }} + requests: + cpu: {{ .Values.gateway.resources.requests.cpu | quote }} + memory: {{ .Values.gateway.resources.requests.memory | quote }} + env: + - name: AMBASSADOR_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: AMBASSADOR_SINGLE_NAMESPACE + value: "true" + - name: SCOUT_DISABLE + value: "1" + - name: "AMBASSADOR_VERIFY_SSL_FALSE" + value: {{ .Values.gateway.insecureDisableSSLVerify | quote }} + - name: AMBASSADOR_ID + value: {{ include "k10.ambassadorId" . }} +{{- if .Values.global.network.enable_ipv6}} + - name: AMBASSADOR_ENVOY_BIND_ADDRESS + value: '::' + - name: AMBASSADOR_DIAGD_BIND_ADDREASS + value: '[::]' +{{- end }} + livenessProbe: + httpGet: + path: /ambassador/v0/check_alive + port: {{ $admin_port }} + initialDelaySeconds: 30 + periodSeconds: 3 + readinessProbe: + httpGet: + path: /ambassador/v0/check_ready + port: {{ $admin_port }} + initialDelaySeconds: 30 + periodSeconds: 3 + restartPolicy: Always +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/grafana-scc.yaml b/charts/kasten/k10/7.0.1401/templates/grafana-scc.yaml new file mode 100644 index 000000000..014d1be46 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/grafana-scc.yaml @@ -0,0 +1,45 @@ +{{- if .Values.scc.create }} +{{- if .Values.grafana.enabled }} +kind: SecurityContextConstraints +apiVersion: security.openshift.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Name }}-grafana +allowPrivilegedContainer: false +allowHostNetwork: false +allowHostDirVolumePlugin: true +allowHostPorts: true +allowHostPID: false +allowHostIPC: false +readOnlyRootFilesystem: false +requiredDropCapabilities: + - KILL + - MKNOD + - SETUID + - SETGID +defaultAddCapabilities: [] +allowedCapabilities: + - CHOWN +priority: 0 +runAsUser: + type: RunAsAny +seLinuxContext: + type: RunAsAny +fsGroup: + type: RunAsAny +supplementalGroups: + type: RunAsAny +seccompProfiles: + - runtime/default +volumes: + - configMap + - downwardAPI + - emptyDir + - persistentVolumeClaim + - projected + - secret +users: + - system:serviceaccount:{{.Release.Namespace}}:{{.Release.Name}}-grafana +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/ingress.yaml b/charts/kasten/k10/7.0.1401/templates/ingress.yaml new file mode 100644 index 000000000..df347477c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ingress.yaml @@ -0,0 +1,73 @@ +{{- $ingressApiIsStable := eq (include "ingress.isStable" .) "true" -}} +{{- $service_port := .Values.gateway.service.externalPort -}} +{{ if .Values.ingress.create }} +{{ include "authEnabled.check" . }} +{{ include "check.ingress.defaultBackend" . }} +apiVersion: {{ template "ingress.apiVersion" . }} +kind: Ingress +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: {{ .Values.ingress.name | default (printf "%s-ingress" .Release.Name) }} + annotations: +{{ include "ingressClassAnnotation" . | indent 4 }} + {{- if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) }} + nginx.ingress.kubernetes.io/secure-backends: "true" + nginx.ingress.kubernetes.io/backend-protocol: HTTPS + {{- end }} + {{- if .Values.ingress.annotations }} +{{ toYaml .Values.ingress.annotations | indent 4 }} + {{- end }} +spec: +{{ include "specIngressClassName" . | indent 2 }} +{{ with .Values.ingress.defaultBackend }} + {{- if or .service.enabled .resource.enabled }} + defaultBackend: + {{- with .service }} + {{- if .enabled }} + service: + name: {{ required "`name` is required in the `ingress.defaultBackend.service`." .name }} + port: + {{- if .port.name }} + name: {{ .port.name }} + {{- else if .port.number }} + number: {{ .port.number }} + {{- end }} + {{- end }} + {{- end }} + {{- with .resource }} + {{- if .enabled }} + resource: + apiGroup: {{ .apiGroup }} + name: {{ required "`name` is required in the `ingress.defaultBackend.resource`." .name }} + kind: {{ required "`kind` is required in the `ingress.defaultBackend.resource`." .kind }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} +{{- if .Values.ingress.tls.enabled }} + tls: + - hosts: + - {{ required "ingress.host value is required for TLS configuration" .Values.ingress.host }} + secretName: {{ .Values.ingress.tls.secretName }} +{{- end }} + rules: + - http: + paths: + - path: /{{ default .Release.Name .Values.ingress.urlPath | trimPrefix "/" | trimSuffix "/" }}/ + pathType: {{ default "ImplementationSpecific" .Values.ingress.pathType }} + backend: + {{- if $ingressApiIsStable }} + service: + name: gateway + port: + number: {{ $service_port }} + {{- else }} + serviceName: gateway + servicePort: {{ $service_port }} + {{- end }} + {{- if .Values.ingress.host }} + host: {{ .Values.ingress.host }} + {{- end }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/templates/k10-config.yaml b/charts/kasten/k10/7.0.1401/templates/k10-config.yaml new file mode 100644 index 000000000..731d29659 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/k10-config.yaml @@ -0,0 +1,355 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-config +data: + DataStoreLogLevel: {{ default "error" | quote }} + DataStoreFileLogLevel: {{ default "" | quote }} + loglevel: {{ .Values.logLevel | quote }} + {{- if .Values.clusterName }} + clustername: {{ quote .Values.clusterName }} + {{- end }} + version: {{ .Chart.AppVersion }} + {{- $capabilities := include "k10.capabilities" . | splitList " " }} + {{- $capabilities_mask := include "k10.capabilities_mask" . | splitList " " }} + {{- if and ( has "mc" $capabilities ) ( not ( has "mc" $capabilities_mask ) ) }} + multiClusterVersion: {{ include "k10.multiClusterVersion" . | quote }} + {{- end }} + modelstoredirname: "//mnt/k10state/kasten-io/" + apiDomain: {{ include "apiDomain" . }} + k10DataStoreDisableCompression: "false" + k10DataStoreParallelUpload: {{ .Values.datastore.parallelUploads | quote }} + k10DataStoreParallelDownload: {{ .Values.datastore.parallelDownloads | quote }} + k10DataStoreGeneralContentCacheSizeMB: {{ include "k10.defaultK10DataStoreGeneralContentCacheSizeMB" . | quote }} + k10DataStoreGeneralMetadataCacheSizeMB: {{ include "k10.defaultK10DataStoreGeneralMetadataCacheSizeMB" . | quote }} + k10DataStoreRestoreContentCacheSizeMB: {{ include "k10.defaultK10DataStoreRestoreContentCacheSizeMB" . | quote }} + k10DataStoreRestoreMetadataCacheSizeMB: {{ include "k10.defaultK10DataStoreRestoreMetadataCacheSizeMB" . | quote }} + K10BackupBufferFileHeadroomFactor: {{ include "k10.defaultK10BackupBufferFileHeadroomFactor" . | quote }} + AWSAssumeRoleDuration: {{ default (include "k10.defaultAssumeRoleDuration" .) .Values.awsConfig.assumeRoleDuration | quote }} + {{- if gt (int .Values.kanister.backupTimeout) 0 }} + K10TimeoutBlueprintBackup: {{ default (include "k10.defaultK10TimeoutBlueprintBackup" .) .Values.kanister.backupTimeout | quote }} + {{- else }} + K10TimeoutBlueprintBackup: {{ default (include "k10.defaultK10TimeoutBlueprintBackup" .) .Values.timeout.blueprintBackup | quote }} + {{- end }} + {{- if gt (int .Values.kanister.restoreTimeout) 0 }} + K10TimeoutBlueprintRestore: {{ default (include "k10.defaultK10TimeoutBlueprintRestore" .) .Values.kanister.restoreTimeout | quote }} + {{- else }} + K10TimeoutBlueprintRestore: {{ default (include "k10.defaultK10TimeoutBlueprintRestore" .) .Values.timeout.blueprintRestore | quote }} + {{- end }} + {{- if gt (int .Values.kanister.deleteTimeout) 0 }} + K10TimeoutBlueprintDelete: {{ default (include "k10.defaultK10TimeoutBlueprintDelete" .) .Values.kanister.deleteTimeout | quote }} + {{- else }} + K10TimeoutBlueprintDelete: {{ default (include "k10.defaultK10TimeoutBlueprintDelete" .) .Values.timeout.blueprintDelete | quote }} + {{- end }} + {{- if gt (int .Values.kanister.hookTimeout) 0 }} + K10TimeoutBlueprintHooks: {{ default (include "k10.defaultK10TimeoutBlueprintHooks" .) .Values.kanister.hookTimeout | quote }} + {{- else }} + K10TimeoutBlueprintHooks: {{ default (include "k10.defaultK10TimeoutBlueprintHooks" .) .Values.timeout.blueprintHooks | quote }} + {{- end }} + {{- if gt (int .Values.kanister.checkRepoTimeout) 0 }} + K10TimeoutCheckRepoPodReady: {{ default (include "k10.defaultK10TimeoutCheckRepoPodReady" .) .Values.kanister.checkRepoTimeout | quote }} + {{- else }} + K10TimeoutCheckRepoPodReady: {{ default (include "k10.defaultK10TimeoutCheckRepoPodReady" .) .Values.timeout.checkRepoPodReady | quote }} + {{- end }} + {{- if gt (int .Values.kanister.statsTimeout) 0 }} + K10TimeoutStatsPodReady: {{ default (include "k10.defaultK10TimeoutStatsPodReady" .) .Values.kanister.statsTimeout | quote }} + {{- else }} + K10TimeoutStatsPodReady: {{ default (include "k10.defaultK10TimeoutStatsPodReady" .) .Values.timeout.statsPodReady | quote }} + {{- end }} + {{- if gt (int .Values.kanister.efsPostRestoreTimeout) 0 }} + K10TimeoutEFSRestorePodReady: {{ default (include "k10.defaultK10TimeoutEFSRestorePodReady" .) .Values.kanister.efsPostRestoreTimeout | quote }} + {{- else }} + K10TimeoutEFSRestorePodReady: {{ default (include "k10.defaultK10TimeoutEFSRestorePodReady" .) .Values.timeout.efsRestorePodReady | quote }} + {{- end }} + {{- if gt (int .Values.kanister.podReadyWaitTimeout) 0 }} + K10TimeoutWorkerPodReady: {{ .Values.kanister.podReadyWaitTimeout | quote }} + {{- else }} + K10TimeoutWorkerPodReady: {{ .Values.timeout.workerPodReady | quote }} + {{- end }} + KanisterManagedDataServicesBlueprintsEnabled: {{ .Values.kanister.managedDataServicesBlueprintsEnabled | quote }} + WorkerPodMetricSidecarEnabled: {{ default .Values.kanisterPodMetricSidecar.enabled .Values.workerPodMetricSidecar.enabled | quote }} + WorkerPodMetricSidecarMetricLifetime: {{ default .Values.kanisterPodMetricSidecar.metricLifetime .Values.workerPodMetricSidecar.metricLifetime | quote }} + WorkerPodPushgatewayMetricsInterval: {{ default .Values.kanisterPodMetricSidecar.pushGatewayInterval .Values.workerPodMetricSidecar.pushGatewayInterval | quote }} +{{- include "workerPodMetricSidecarResources" . | indent 2 }} + KanisterToolsImage: {{ include "get.kanisterToolsImage" . | quote }} + K10MutatingWebhookTLSCertDir: "/etc/ssl/certs/webhook" + + {{- if gt (int .Values.limiter.concurrentSnapConversions) 0 }} + K10LimiterSnapshotExportsPerAction: {{ default (include "k10.defaultK10LimiterSnapshotExportsPerAction" .) .Values.limiter.concurrentSnapConversions | quote }} + {{- else }} + K10LimiterSnapshotExportsPerAction: {{ default (include "k10.defaultK10LimiterSnapshotExportsPerAction" .) .Values.limiter.snapshotExportsPerAction | quote }} + {{- end }} + {{- if gt (int .Values.limiter.genericVolumeSnapshots) 0 }} + K10LimiterGenericVolumeBackupsPerCluster: {{ default (include "k10.defaultK10LimiterGenericVolumeBackupsPerCluster" .) .Values.limiter.genericVolumeSnapshots | quote }} + {{- else }} + K10LimiterGenericVolumeBackupsPerCluster: {{ default (include "k10.defaultK10LimiterGenericVolumeBackupsPerCluster" .) .Values.limiter.genericVolumeBackupsPerCluster | quote }} + {{- end }} + {{- if gt (int .Values.limiter.genericVolumeCopies) 0 }} + K10LimiterSnapshotExportsPerCluster: {{ default (include "k10.defaultK10LimiterSnapshotExportsPerCluster" .) .Values.limiter.genericVolumeCopies | quote }} + {{- else }} + K10LimiterSnapshotExportsPerCluster: {{ default (include "k10.defaultK10LimiterSnapshotExportsPerCluster" .) .Values.limiter.snapshotExportsPerCluster | quote }} + {{- end }} + {{- if gt (int .Values.limiter.genericVolumeRestores) 0 }} + K10LimiterVolumeRestoresPerCluster: {{ default (include "k10.defaultK10LimiterVolumeRestoresPerCluster" .) .Values.limiter.genericVolumeRestores | quote }} + {{- else }} + K10LimiterVolumeRestoresPerCluster: {{ default (include "k10.defaultK10LimiterVolumeRestoresPerCluster" .) .Values.limiter.volumeRestoresPerCluster | quote }} + {{- end }} + {{- if gt (int .Values.limiter.csiSnapshots) 0 }} + K10LimiterCsiSnapshotsPerCluster: {{ default (include "k10.defaultK10LimiterCsiSnapshotsPerCluster" .) .Values.limiter.csiSnapshots | quote }} + {{- else }} + K10LimiterCsiSnapshotsPerCluster: {{ default (include "k10.defaultK10LimiterCsiSnapshotsPerCluster" .) .Values.limiter.csiSnapshotsPerCluster | quote }} + {{- end }} + {{- if gt (int .Values.limiter.providerSnapshots) 0 }} + K10LimiterDirectSnapshotsPerCluster: {{ default (include "k10.defaultK10LimiterDirectSnapshotsPerCluster" .) .Values.limiter.providerSnapshots | quote }} + {{- else }} + K10LimiterDirectSnapshotsPerCluster: {{ default (include "k10.defaultK10LimiterDirectSnapshotsPerCluster" .) .Values.limiter.directSnapshotsPerCluster | quote }} + {{- end }} + {{- if gt (int .Values.limiter.imageCopies) 0 }} + K10LimiterImageCopiesPerCluster: {{ default (include "k10.defaultK10LimiterImageCopiesPerCluster" .) .Values.limiter.imageCopies | quote }} + {{- else }} + K10LimiterImageCopiesPerCluster: {{ default (include "k10.defaultK10LimiterImageCopiesPerCluster" .) .Values.limiter.imageCopiesPerCluster | quote }} + {{- end }} + K10LimiterWorkloadSnapshotsPerAction: {{ default (include "k10.defaultK10LimiterWorkloadSnapshotsPerAction" .) .Values.limiter.workloadSnapshotsPerAction | quote }} + {{- if gt (int .Values.services.executor.workerCount) 0 }} + K10LimiterExecutorThreads: {{ default (include "k10.defaultK10LimiterExecutorThreads" .) .Values.services.executor.workerCount | quote }} + {{- else }} + K10LimiterExecutorThreads: {{ default (include "k10.defaultK10LimiterExecutorThreads" .) .Values.limiter.executorThreads | quote }} + {{- end }} + {{- if gt (int .Values.services.executor.maxConcurrentRestoreCsiSnapshots) 0 }} + K10LimiterCsiSnapshotRestoresPerAction: {{ default (include "k10.defaultK10LimiterCsiSnapshotRestoresPerAction" .) .Values.services.executor.maxConcurrentRestoreCsiSnapshots | quote }} + {{- else }} + K10LimiterCsiSnapshotRestoresPerAction: {{ default (include "k10.defaultK10LimiterCsiSnapshotRestoresPerAction" .) .Values.limiter.csiSnapshotRestoresPerAction | quote }} + {{- end }} + {{- if gt (int .Values.services.executor.maxConcurrentRestoreGenericVolumeSnapshots) 0 }} + K10LimiterVolumeRestoresPerAction: {{ default (include "k10.defaultK10LimiterVolumeRestoresPerAction" .) .Values.services.executor.maxConcurrentRestoreGenericVolumeSnapshots | quote }} + {{- else }} + K10LimiterVolumeRestoresPerAction: {{ default (include "k10.defaultK10LimiterVolumeRestoresPerAction" .) .Values.limiter.volumeRestoresPerAction | quote }} + {{- end }} + {{- if gt (int .Values.services.executor.maxConcurrentRestoreWorkloads) 0 }} + K10LimiterWorkloadRestoresPerAction: {{ default (include "k10.defaultK10LimiterWorkloadRestoresPerAction" .) .Values.services.executor.maxConcurrentRestoreWorkloads | quote }} + {{- else }} + K10LimiterWorkloadRestoresPerAction: {{ default (include "k10.defaultK10LimiterWorkloadRestoresPerAction" .) .Values.limiter.workloadRestoresPerAction | quote }} + {{- end }} + + K10GCDaemonPeriod: {{ default (include "k10.defaultK10GCDaemonPeriod" .) .Values.garbagecollector.daemonPeriod | quote }} + K10GCKeepMaxActions: {{ default (include "k10.defaultK10GCKeepMaxActions" .) .Values.garbagecollector.keepMaxActions | quote }} + K10GCActionsEnabled: {{ default (include "k10.defaultK10GCActionsEnabled" .) .Values.garbagecollector.actions.enabled | quote }} + + K10EphemeralPVCOverhead: {{ .Values.ephemeralPVCOverhead | quote }} + + K10PersistenceStorageClass: {{ .Values.global.persistence.storageClass | quote }} + + K10DefaultPriorityClassName: {{ default (include "k10.defaultK10DefaultPriorityClassName" .) .Values.defaultPriorityClassName | quote }} + {{- if .Values.global.podLabels }} + K10CustomPodLabels: {{ include "k10.globalPodLabelsJson" . | quote }} + {{- end }} + {{- if .Values.global.podAnnotations }} + K10CustomPodAnnotations: {{ include "k10.globalPodAnnotationsJson" . | quote }} + {{- end }} + + kubeVirtVMsUnFreezeTimeout: {{ default (include "k10.defaultKubeVirtVMsUnfreezeTimeout" .) .Values.kubeVirtVMs.snapshot.unfreezeTimeout | quote }} + + {{- if not (quote .Values.maxJobWaitDuration | empty) }} + K10TimeoutJobWait: {{ .Values.maxJobWaitDuration | quote }} + {{- else }} + K10TimeoutJobWait: {{ .Values.timeout.jobWait | quote }} + {{- end }} + + quickDisasterRecoveryEnabled: {{ .Values.kastenDisasterRecovery.quickMode.enabled | quote }} + + {{- if .Values.forceRootInKanisterHooks }} + K10ForceRootInBlueprintActions: {{ .Values.forceRootInKanisterHooks| quote }} + {{- else }} + K10ForceRootInBlueprintActions: {{ .Values.forceRootInBlueprintActions | quote }} + {{- end }} + + workerPodResourcesCRDEnabled: {{ .Values.workerPodCRDs.enabled | quote }} +{{- include "workerPodResourcesCRD" . | indent 2 }} + + {{- if .Values.awsConfig.efsBackupVaultName }} + efsBackupVaultName: {{ quote .Values.awsConfig.efsBackupVaultName }} + {{- end }} + + {{- if .Values.excludedApps }} + excludedApps: '{{ join "," .Values.excludedApps }}' + {{- end }} + + {{- if .Values.vmWare.taskTimeoutMin }} + vmWareTaskTimeoutMin: {{ quote .Values.vmWare.taskTimeoutMin }} + {{- end }} + +{{- include "get.kanisterPodCustomLabels" . | indent 2}} +{{- include "get.kanisterPodCustomAnnotations" . | indent 2}} + + {{- if .Values.kanisterFunctionVersion }} + kanisterFunctionVersion: {{ .Values.kanisterFunctionVersion | quote }} + {{- else }} + kanisterFunctionVersion: {{ quote "v1.0.0-alpha" }} + {{- end }} +{{- include "kanisterToolsResources" . | indent 2 }} +{{- include "get.gvsActivationToken" . | indent 2 }} + + {{- if .Values.genericStorageBackup.overridepubkey }} + overridePublicKeyForGVS: {{ .Values.genericStorageBackup.overridepubkey | quote }} + {{- end }} + + {{- with (include "k10.fluentbitEndpoint" .) }} + fluentbitEndpoint: {{ . | quote }} + {{- end }} + +{{ if .Values.features }} +--- +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-features +data: +{{ include "k10.features" . | indent 2}} +{{ end }} +{{ if .Values.auth.openshift.enabled }} +--- +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-dex + namespace: {{ .Release.Namespace }} +data: + config.yaml: | + issuer: {{ printf "%s/dex" (trimSuffix "/" .Values.auth.openshift.dashboardURL) }} + storage: + type: memory + web: + http: 0.0.0.0:8080 + logger: + level: info + format: text + connectors: + - type: openshift + id: openshift + name: OpenShift + config: + issuer: {{ .Values.auth.openshift.openshiftURL }} + clientID: {{ printf "system:serviceaccount:%s:%s" .Release.Namespace (include "get.openshiftServiceAccountName" .) }} + clientSecret: {{ printf "{{ getenv \"%s\" }}" (include "k10.openShiftClientSecretEnvVar" . ) }} + redirectURI: {{ printf "%s/dex/callback" (trimSuffix "/" .Values.auth.openshift.dashboardURL) }} + insecureCA: {{ .Values.auth.openshift.insecureCA }} +{{- if and (eq (include "check.cacertconfigmap" .) "false") .Values.auth.openshift.useServiceAccountCA }} + rootCA: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt +{{- end }} + oauth2: + skipApprovalScreen: true + staticClients: + - name: 'K10' + id: kasten + secret: kastensecret + redirectURIs: + - {{ printf "%s/auth-svc/v0/oidc/redirect" (trimSuffix "/" .Values.auth.openshift.dashboardURL) }} +{{ end }} +{{ if .Values.auth.ldap.enabled }} +--- +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-dex + namespace: {{ .Release.Namespace }} +data: + config.yaml: | + issuer: {{ printf "%s/dex" (trimSuffix "/" .Values.auth.ldap.dashboardURL) }} + storage: + type: memory + web: + http: 0.0.0.0:8080 + frontend: + dir: {{ include "k10.dexFrontendDir" . }} + theme: custom + logoURL: theme/kasten-logo.svg + logger: + level: info + format: text + connectors: + - type: ldap + id: ldap + name: LDAP + config: + host: {{ .Values.auth.ldap.host }} + insecureNoSSL: {{ .Values.auth.ldap.insecureNoSSL }} + insecureSkipVerify: {{ .Values.auth.ldap.insecureSkipVerifySSL }} + startTLS: {{ .Values.auth.ldap.startTLS }} + bindDN: {{ .Values.auth.ldap.bindDN }} + bindPW: BIND_PASSWORD_PLACEHOLDER + userSearch: + baseDN: {{ .Values.auth.ldap.userSearch.baseDN }} + filter: {{ .Values.auth.ldap.userSearch.filter }} + username: {{ .Values.auth.ldap.userSearch.username }} + idAttr: {{ .Values.auth.ldap.userSearch.idAttr }} + emailAttr: {{ .Values.auth.ldap.userSearch.emailAttr }} + nameAttr: {{ .Values.auth.ldap.userSearch.nameAttr }} + preferredUsernameAttr: {{ .Values.auth.ldap.userSearch.preferredUsernameAttr }} + groupSearch: + baseDN: {{ .Values.auth.ldap.groupSearch.baseDN }} + filter: {{ .Values.auth.ldap.groupSearch.filter }} + nameAttr: {{ .Values.auth.ldap.groupSearch.nameAttr }} +{{- with .Values.auth.ldap.groupSearch.userMatchers }} + userMatchers: +{{ toYaml . | indent 10 }} +{{- end }} + oauth2: + skipApprovalScreen: true + staticClients: + - name: 'K10' + id: kasten + secret: kastensecret + redirectURIs: + - {{ printf "%s/auth-svc/v0/oidc/redirect" (trimSuffix "/" .Values.auth.ldap.dashboardURL) }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: k10-logos-dex + namespace: {{ .Release.Namespace }} +binaryData: + {{- $files := .Files }} + {{- range tuple "files/favicon.png" "files/kasten-logo.svg" "files/styles.css" }} + {{ trimPrefix "files/" . }}: |- + {{ $files.Get . | b64enc }} + {{- end }} +{{ end }} +{{ if (include "k10.capability.gateway" $) }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: k10-gateway + namespace: {{ .Release.Namespace }} +data: + {{ include "k10.gatewayPrefixVarName" . }}: {{ include "k10.prefixPath" . }} + {{ include "k10.gatewayGrafanaSvcVarName" . }}: {{ printf "%s-grafana" .Release.Name }} + + {{- if .Values.gateway.requestHeaders }} + {{ include "k10.gatewayRequestHeadersVarName" .}}: {{ (.Values.gateway.requestHeaders | default list) | join " " }} + {{- end }} + + {{- if .Values.gateway.authHeaders }} + {{ include "k10.gatewayAuthHeadersVarName" .}}: {{ (.Values.gateway.authHeaders | default list) | join " " }} + {{- end }} + + {{- if .Values.gateway.service.internalPort }} + {{ include "k10.gatewayPortVarName" .}}: {{ .Values.gateway.service.internalPort | quote }} + {{- end }} + + {{- if or .Values.secrets.tlsSecret (and .Values.secrets.apiTlsCrt .Values.secrets.apiTlsKey) }} + {{ include "k10.gatewayTLSCertFile" . }}: /etc/tls/tls.crt + {{ include "k10.gatewayTLSKeyFile" . }}: /etc/tls/tls.key + {{- end }} + +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/templates/k10-eula.yaml b/charts/kasten/k10/7.0.1401/templates/k10-eula.yaml new file mode 100644 index 000000000..21e251d6c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/k10-eula.yaml @@ -0,0 +1,21 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-eula +data: + text: {{ .Files.Get "eula.txt" | quote }} +--- +{{ if .Values.eula.accept }} +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-eula-info +data: +{{ include "k10.eula.fields" . | indent 2 }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/templates/k10-scc.yaml b/charts/kasten/k10/7.0.1401/templates/k10-scc.yaml new file mode 100644 index 000000000..12a449f6f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/k10-scc.yaml @@ -0,0 +1,46 @@ +{{- if .Values.scc.create }} +kind: SecurityContextConstraints +apiVersion: security.openshift.io/v1 +metadata: + name: {{ .Release.Name }}-scc + labels: +{{ include "helm.labels" . | indent 4 }} +allowHostDirVolumePlugin: false +allowHostIPC: false +allowHostNetwork: false +allowHostPID: false +allowHostPorts: false +allowPrivilegeEscalation: false +allowPrivilegedContainer: false +allowedCapabilities: + - CHOWN + - FOWNER + - DAC_OVERRIDE +defaultAddCapabilities: + - CHOWN + - FOWNER + - DAC_OVERRIDE +fsGroup: + type: RunAsAny +priority: {{ .Values.scc.priority }} +readOnlyRootFilesystem: false +requiredDropCapabilities: + - ALL +runAsUser: + type: RunAsAny +seLinuxContext: + type: RunAsAny +supplementalGroups: + type: RunAsAny +seccompProfiles: + - runtime/default +users: + - system:serviceaccount:{{.Release.Namespace}}:{{ template "serviceAccountName" . }} +volumes: + - configMap + - downwardAPI + - emptyDir + - persistentVolumeClaim + - projected + - secret +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/kopia-tls-certs.yaml b/charts/kasten/k10/7.0.1401/templates/kopia-tls-certs.yaml new file mode 100644 index 000000000..ac0635f51 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/kopia-tls-certs.yaml @@ -0,0 +1,33 @@ +# alternate names of the services. This renders to: [ component-svc.namespace, component-svc.namespace.svc ] +{{- $altNamesKopia := list ( printf "%s-svc.%s" "data-mover" .Release.Namespace ) ( printf "%s-svc.%s.svc" "data-mover" .Release.Namespace ) }} +# generate ca cert with 365 days of validity +{{- $caKopia := genCA ( printf "%s-svc-ca" "data-mover" ) 365 }} +# generate cert with CN="component-svc", SAN=$altNames and with 365 days of validity +{{- $certKopia := genSignedCert ( printf "%s-svc" "data-mover" ) nil $altNamesKopia 365 $caKopia }} +apiVersion: v1 +kind: Secret +type: Opaque +metadata: + name: kopia-tls-cert + labels: +{{ include "helm.labels" . | indent 4 }} +{{- if .Values.global.rhMarketPlace }} + annotations: + "helm.sh/hook": "pre-install" +{{- end }} +data: + tls.crt: {{ $certKopia.Cert | b64enc }} +--- +apiVersion: v1 +kind: Secret +type: Opaque +metadata: + name: kopia-tls-key + labels: +{{ include "helm.labels" . | indent 4 }} +{{- if .Values.global.rhMarketPlace }} + annotations: + "helm.sh/hook": "pre-install" +{{- end }} +data: + tls.key: {{ $certKopia.Key | b64enc }} diff --git a/charts/kasten/k10/7.0.1401/templates/license.yaml b/charts/kasten/k10/7.0.1401/templates/license.yaml new file mode 100644 index 000000000..f409fb7e5 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/license.yaml @@ -0,0 +1,25 @@ +{{- if not ( or ( .Values.license ) ( .Values.metering.awsMarketplace ) ( .Values.metering.awsManagedLicense ) ( .Values.metering.licenseConfigSecretName ) ) }} +{{- if .Files.Get "triallicense" }} +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-trial-license +type: Opaque +data: + license: {{ print (.Files.Get "triallicense") }} +{{- end }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-license +type: Opaque +data: + license: {{ include "k10.getlicense" . }} diff --git a/charts/kasten/k10/7.0.1401/templates/mc.yaml b/charts/kasten/k10/7.0.1401/templates/mc.yaml new file mode 100644 index 000000000..2c23f94ae --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/mc.yaml @@ -0,0 +1,6 @@ +{{- if not .Values.multicluster.enabled -}} + {{- $clusterInfo := lookup "v1" "Secret" .Release.Namespace "mc-cluster-info" -}} + {{- if $clusterInfo -}} + {{- fail "WARNING: Multi-cluster features must remain enabled as long as this cluster is connected to a multi-cluster system.\nEither disconnect this cluster from the multi-cluster system or use multicluster.enabled=true to enable multi-cluster features." -}} + {{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/mutatingwebhook.yaml b/charts/kasten/k10/7.0.1401/templates/mutatingwebhook.yaml new file mode 100644 index 000000000..729df5865 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/mutatingwebhook.yaml @@ -0,0 +1,51 @@ +{{- if .Values.injectKanisterSidecar.enabled -}} +# alternate names of the services. This renders to: [ component-svc.namespace, component-svc.namespace.svc ] +{{- $altNames := list ( printf "%s-svc.%s" "controllermanager" .Release.Namespace ) ( printf "%s-svc.%s.svc" "controllermanager" .Release.Namespace ) }} +# generate ca cert with 365 days of validity +{{- $ca := genCA ( printf "%s-svc-ca" "controllermanager" ) 365 }} +# generate cert with CN="component-svc", SAN=$altNames and with 365 days of validity +{{- $cert := genSignedCert ( printf "%s-svc" "controllermanager" ) nil $altNames 365 $ca }} +apiVersion: v1 +kind: Secret +type: kubernetes.io/tls +metadata: + name: controllermanager-certs + labels: +{{ include "helm.labels" . | indent 4 }} +data: + tls.crt: {{ $cert.Cert | b64enc }} + tls.key: {{ $cert.Key | b64enc }} +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-sidecar-injector +webhooks: +- name: k10-sidecar-injector.kasten.io + admissionReviewVersions: ["v1", "v1beta1"] + failurePolicy: Ignore + sideEffects: None + clientConfig: + service: + name: controllermanager-svc + namespace: {{ .Release.Namespace }} + path: "/k10/mutate" + port: 443 + caBundle: {{ b64enc $ca.Cert }} + rules: + - operations: ["CREATE", "UPDATE"] + apiGroups: ["*"] + apiVersions: ["v1"] + resources: ["deployments", "statefulsets", "deploymentconfigs"] +{{- if .Values.injectKanisterSidecar.namespaceSelector }} + namespaceSelector: +{{ toYaml .Values.injectKanisterSidecar.namespaceSelector | indent 4 }} +{{- end }} +{{- if .Values.injectKanisterSidecar.objectSelector }} + objectSelector: +{{ toYaml .Values.injectKanisterSidecar.objectSelector | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1401/templates/networkpolicy.yaml new file mode 100644 index 000000000..0f7629580 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/networkpolicy.yaml @@ -0,0 +1,299 @@ +{{- $admin_port := default 8877 .Values.service.gatewayAdminPort -}} +{{- $mutating_webhook_port := default 8080 .Values.injectKanisterSidecar.webhookServer.port -}} +{{- if .Values.networkPolicy.create }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: default-deny + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: {} + policyTypes: + - Ingress +{{- if eq (include "k10.isOpenShift" .) "true" }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-openshift-console-access + namespace: {{ .Release.Namespace }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + ingress: + - from: + - namespaceSelector: + matchLabels: + kubernetes.io/metadata.name: openshift-console +{{- end }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: access-k10-services + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + ingress: + - from: + - podSelector: + matchLabels: + access-k10-services: allowed + ports: + - protocol: TCP + port: {{ .Values.service.internalPort }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: cross-services-allow + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + ingress: + - from: + - podSelector: + matchLabels: + release: {{ .Release.Name }} + ports: + - protocol: TCP + port: {{ .Values.service.internalPort }} +--- +{{/* TODO: Consider a flag to turn this off. */}} +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-gateway-to-mc-external + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + component: controllermanager + release: {{ .Release.Name }} + ingress: + - from: + - podSelector: + matchLabels: + service: gateway + release: {{ .Release.Name }} + ports: + - protocol: TCP + port: {{ include "k10.mcExternalPort" nil }} +{{- if .Values.logging.internal }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: logging-allow-internal + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + run: logging-svc + ingress: + - from: + - podSelector: + matchLabels: + release: {{ .Release.Name }} + ports: + # Logging input port + - protocol: TCP + port: 24224 + - protocol: TCP + port: 24225 +{{- end }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-external + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + service: gateway + release: {{ .Release.Name }} + ingress: + - from: [] + ports: + - protocol: TCP + port: {{ .Values.gateway.service.internalPort | default 8000 }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-all-api + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + run: aggregatedapis-svc + release: {{ .Release.Name }} + ingress: + - from: + ports: + - protocol: TCP + port: {{ .Values.service.aggregatedApiPort }} +{{- if .Values.gateway.exposeAdminPort }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-gateway-admin + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + service: gateway + ingress: + - from: + - podSelector: + matchLabels: + app: prometheus + component: server + release: {{ .Release.Name }} + ports: + - protocol: TCP + port: {{ $admin_port }} +{{- end -}} +{{- if or .Values.workerPodMetricSidecar.enabled .Values.kanisterPodMetricSidecar.enabled }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-metrics-kanister-pods + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + run: metering-svc + ingress: + - from: + - podSelector: + matchLabels: + createdBy: kanister + ports: + - protocol: TCP + port: {{ .Values.service.internalPort }} +{{- end -}} +{{- if .Values.injectKanisterSidecar.enabled }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: allow-mutating-webhook + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + run: controllermanager-svc + ingress: + - from: + ports: + - protocol: TCP + port: {{ $mutating_webhook_port }} +{{- end -}} +{{- if eq (include "check.dexAuth" .) "true" }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: gateway-dex-allow + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + run: auth-svc + ingress: + - from: + - podSelector: + matchLabels: + service: gateway + release: {{ .Release.Name }} + ports: + - protocol: TCP + port: 8080 +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: auth-dex-allow + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ .Release.Name }} + run: auth-svc + ingress: + - from: + - podSelector: + matchLabels: + run: auth-svc + release: {{ .Release.Name }} + ports: + - protocol: TCP + port: 8080 +{{- end -}} +{{- $mainCtx := . }} +{{- $colocatedList := include "get.enabledColocatedSvcList" . | fromYaml }} +{{- range $primary, $secondaryList := $colocatedList }} +--- +kind: NetworkPolicy +apiVersion: networking.k8s.io/v1 +metadata: + name: {{ $primary }}-svc-allow-secondary-services + namespace: {{ $mainCtx.Release.Namespace }} + labels: +{{ include "helm.labels" $mainCtx | indent 4 }} +spec: + podSelector: + matchLabels: + release: {{ $mainCtx.Release.Name }} + run: {{ $primary }}-svc + ingress: + - from: + - podSelector: + matchLabels: + release: {{ $mainCtx.Release.Name }} + ports: + {{- range $skip, $secondary := $secondaryList }} + {{- $colocConfig := index (include "get.enabledColocatedServices" $mainCtx | fromYaml) $secondary }} + - protocol: TCP + port: {{ $colocConfig.port }} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-ca-cert-extract-hook.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-ca-cert-extract-hook.yaml new file mode 100644 index 000000000..73f39bb9c --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-ca-cert-extract-hook.yaml @@ -0,0 +1,218 @@ +{{- if (include "k10.ocpcacertsautoextraction" .) -}} +{{- if or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath }} +--- +apiVersion: v1 +kind: Secret +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: {{ .Release.Name }}-extract-ocp-ca-cert-dockerconfig + namespace: {{ .Release.Namespace }} +type: kubernetes.io/dockerconfigjson +data: + .dockerconfigjson: {{ or .Values.secrets.dockerConfig ( .Values.secrets.dockerConfigPath | b64enc ) }} +{{- end }} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: openshift-cluster-config-reader +rules: + - apiGroups: ["config.openshift.io"] + resources: ["proxies", "apiservers"] + verbs: ["get", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: openshift-config-reader + namespace: openshift-config +rules: + - apiGroups: [""] + resources: ["configmaps", "secrets"] + verbs: ["get", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: openshift-ingress-operator-reader + namespace: openshift-ingress-operator +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: openshift-kube-apiserver-reader + namespace: openshift-kube-apiserver +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "1" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: {{ .Release.Namespace }}-configmaps-editor + namespace: {{ .Release.Namespace }} +rules: + - apiGroups: [""] + resources: ["configmaps"] + verbs: ["create", "get", "list", "watch", "patch", "update"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "2" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: read-openshift-cluster-config +subjects: + - kind: ServiceAccount + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +roleRef: + kind: ClusterRole + name: openshift-cluster-config-reader + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "2" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: read-openshift-config + namespace: openshift-config +subjects: + - kind: ServiceAccount + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: openshift-config-reader + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "2" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: read-openshift-ingress-operator + namespace: openshift-ingress-operator +subjects: + - kind: ServiceAccount + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: openshift-ingress-operator-reader + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "2" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: read-openshift-kube-apiserver + namespace: openshift-kube-apiserver +subjects: + - kind: ServiceAccount + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: openshift-kube-apiserver-reader + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "2" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + name: edit-{{ .Release.Namespace }}-configmaps + namespace: {{ .Release.Namespace }} +subjects: + - kind: ServiceAccount + name: {{ .Release.Name }}-ocp-ca-cert-extractor + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: {{ .Release.Namespace }}-configmaps-editor + apiGroup: rbac.authorization.k8s.io +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ .Release.Name }}-extract-ocp-ca-cert-job + labels: +{{ include "helm.labels" . | indent 4 }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded,hook-failed + "helm.sh/hook-weight": "3" +spec: + template: + metadata: + name: {{ .Release.Name }}-extract-ocp-ca-cert-job + labels: +{{ include "helm.labels" . | indent 8 }} + spec: + restartPolicy: Never + serviceAccountName: {{ .Release.Name }}-ocp-ca-cert-extractor + containers: + - name: {{ .Release.Name }}-extract-ocp-ca-cert-job + image: {{ include "k10.k10ToolsImage" . }} + command: ["./k10tools", "openshift", "extract-certificates"] + args: ["-n", "{{ .Release.Namespace }}", "--release-name", "{{ .Release.Name }}", "--ca-cert-configmap-name", "{{ .Values.cacertconfigmap.name }}"] + {{- if or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath }} + imagePullSecrets: + - name: {{ .Release.Name }}-extract-ocp-ca-cert-dockerconfig + {{- else if .Values.global.imagePullSecret }} + imagePullSecrets: + - name: {{ .Values.global.imagePullSecret }} + {{- end }} + backoffLimit: 0 +{{ end }} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-consoleplugin.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-consoleplugin.yaml new file mode 100644 index 000000000..845d9667d --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-consoleplugin.yaml @@ -0,0 +1,27 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: console.openshift.io/v1 +kind: ConsolePlugin +metadata: + name: {{ template "k10.openShiftConsolePluginName" . }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginName" . }} +spec: + displayName: Veeam Kasten Plugin + backend: + type: Service + service: + name: {{ template "k10.openShiftConsolePluginName" . }} + namespace: {{ .Release.Namespace }} + port: 9443 + basePath: / + proxy: + - alias: dashboardbff + authorization: UserToken + endpoint: + service: + name: {{ template "k10.openShiftConsolePluginProxyName" . }} + namespace: {{ .Release.Namespace }} + port: 443 + type: Service +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-configmap.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-configmap.yaml new file mode 100644 index 000000000..1fc058271 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-configmap.yaml @@ -0,0 +1,28 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "k10.openShiftConsolePluginConfigMapName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginName" . }} +data: + nginx.conf: | + pid /var/run/nginx/nginx.pid; + error_log /dev/stdout info; + events {} + http { + access_log /dev/stdout; + include /etc/nginx/mime.types; + default_type application/octet-stream; + keepalive_timeout 65; + server { + listen 9443 ssl; + listen [::]:9443 ssl; + ssl_certificate /var/cert/tls.crt; + ssl_certificate_key /var/cert/tls.key; + root /ocpconsoleplugin; + } + } +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-deployment.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-deployment.yaml new file mode 100644 index 000000000..3be16b1c9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-deployment.yaml @@ -0,0 +1,74 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "k10.openShiftConsolePluginName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginName" . }} + app.openshift.io/runtime-namespace: {{ .Release.Namespace }} +spec: + replicas: 2 + selector: + matchLabels: +{{ include "helm.labels" . | indent 6 }} + component: {{ template "k10.openShiftConsolePluginName" . }} + template: + metadata: + {{- if .Values.scc.create }} + annotations: + openshift.io/required-scc: {{ .Release.Name }}-scc + {{- end }} + labels: +{{ include "helm.labels" . | indent 8 }} + component: {{ template "k10.openShiftConsolePluginName" . }} + spec: + containers: + - name: {{ template "k10.openShiftConsolePluginName" . }} + image: {{ include "k10.ocpConsolePluginImage" . }} + ports: + - containerPort: 9443 + protocol: TCP + imagePullPolicy: Always + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: false + capabilities: + drop: ["ALL"] + resources: + requests: + cpu: 10m + memory: 50Mi + volumeMounts: + - name: {{ template "k10.openShiftConsolePluginTLSCertName" . }} + readOnly: true + mountPath: /var/cert + - name: {{ template "k10.openShiftConsolePluginConfigMapName" . }} + readOnly: true + mountPath: /etc/nginx/nginx.conf + subPath: nginx.conf + volumes: + - name: {{ template "k10.openShiftConsolePluginTLSCertName" . }} + secret: + secretName: {{ template "k10.openShiftConsolePluginTLSCertName" . }} + defaultMode: 420 + - name: {{ template "k10.openShiftConsolePluginConfigMapName" . }} + configMap: + name: {{ template "k10.openShiftConsolePluginConfigMapName" . }} + defaultMode: 420 + restartPolicy: Always + dnsPolicy: ClusterFirst + securityContext: + fsGroup: 1000 + runAsNonRoot: true + runAsUser: 1000 + seccompProfile: + type: RuntimeDefault + serviceAccountName: {{ template "serviceAccountName" . }} + strategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 25% + maxSurge: 25% +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-configmap.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-configmap.yaml new file mode 100644 index 000000000..3276a3265 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-configmap.yaml @@ -0,0 +1,36 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "k10.openShiftConsolePluginProxyConfigMapName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} +data: + nginx.conf: | + pid /var/run/nginx/nginx.pid; + error_log /dev/stdout info; + events { + worker_connections 1024; + } + http { + access_log /dev/stdout; + server { + listen 8080; + server_name {{ template "k10.openShiftConsolePluginProxyName" . }}.dashboardbff; + location / { + proxy_pass http://dashboardbff-svc.{{ .Release.Namespace }}:8000; + } + } + server { + listen 9443 ssl; + server_name {{ template "k10.openShiftConsolePluginProxyName" . }}.dashboardbff; + ssl_certificate /etc/nginx/ssl/tls.crt; + ssl_certificate_key /etc/nginx/ssl/tls.key; + location / { + proxy_pass http://dashboardbff-svc.{{ .Release.Namespace }}:8000; + } + } + } +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-deployment.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-deployment.yaml new file mode 100644 index 000000000..50f15463a --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-deployment.yaml @@ -0,0 +1,67 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "k10.openShiftConsolePluginProxyName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} +spec: + replicas: 1 + selector: + matchLabels: +{{ include "helm.labels" . | indent 6 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} + template: + metadata: + {{- if .Values.scc.create }} + annotations: + openshift.io/required-scc: {{ .Release.Name }}-scc + {{- end }} + labels: +{{ include "helm.labels" . | indent 8 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} + spec: + containers: + - image: {{ include "k10.ocpConsolePluginImage" . }} + imagePullPolicy: Always + securityContext: + allowPrivilegeEscalation: false + readOnlyRootFilesystem: false + capabilities: + drop: ["ALL"] + name: nginx + ports: + - containerPort: 8080 + name: http + protocol: TCP + - containerPort: 9443 + name: https + protocol: TCP + resources: + requests: + cpu: 10m + memory: 50Mi + volumeMounts: + - mountPath: /etc/nginx + name: {{ template "k10.openShiftConsolePluginProxyConfigMapName" . }} + - mountPath: /etc/nginx/ssl + name: {{ template "k10.openShiftConsolePluginProxyTLSCertName" . }} + volumes: + - name: {{ template "k10.openShiftConsolePluginProxyConfigMapName" . }} + configMap: + defaultMode: 420 + name: {{ template "k10.openShiftConsolePluginProxyConfigMapName" . }} + - name: {{ template "k10.openShiftConsolePluginProxyTLSCertName" . }} + secret: + defaultMode: 420 + secretName: {{ template "k10.openShiftConsolePluginProxyTLSCertName" . }} + securityContext: + fsGroup: 1000 + runAsNonRoot: true + runAsUser: 1000 + seccompProfile: + type: RuntimeDefault + serviceAccountName: {{ template "serviceAccountName" . }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-service.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-service.yaml new file mode 100644 index 000000000..e1b421673 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-proxy-service.yaml @@ -0,0 +1,25 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: v1 +kind: Service +metadata: + annotations: + service.beta.openshift.io/serving-cert-secret-name: {{ template "k10.openShiftConsolePluginProxyTLSCertName" . }} + name: {{ template "k10.openShiftConsolePluginProxyName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} +spec: + selector: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginProxyName" . }} + ports: + - name: http + port: 80 + protocol: TCP + targetPort: http + - name: https + port: 443 + protocol: TCP + targetPort: https +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/ocp-plugin-service.yaml b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-service.yaml new file mode 100644 index 000000000..59add4bc4 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/ocp-plugin-service.yaml @@ -0,0 +1,23 @@ +{{- if eq (include "k10.isOpenShift" .) "true" -}} +apiVersion: v1 +kind: Service +metadata: + annotations: + service.alpha.openshift.io/serving-cert-secret-name: {{ template "k10.openShiftConsolePluginTLSCertName" . }} + name: {{ template "k10.openShiftConsolePluginName" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginName" . }} +spec: + ports: + - name: tcp + protocol: TCP + port: 9443 + targetPort: 9443 + selector: +{{ include "helm.labels" . | indent 4 }} + component: {{ template "k10.openShiftConsolePluginName" . }} + type: ClusterIP + sessionAffinity: None +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/prometheus-configmap.yaml b/charts/kasten/k10/7.0.1401/templates/prometheus-configmap.yaml new file mode 100644 index 000000000..4e1e24f9e --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/prometheus-configmap.yaml @@ -0,0 +1,97 @@ +{{ include "check.validatePrometheusConfig" .}} +{{- if .Values.prometheus.server.enabled -}} +{{- $cluster_domain := "" -}} +{{- with .Values.cluster.domainName -}} + {{- $cluster_domain = printf ".%s" . -}} +{{- end -}} +{{- $rbac := .Values.prometheus.rbac.create -}} +kind: ConfigMap +apiVersion: v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: {{ .Release.Name }}-{{ .Values.prometheus.server.configMapOverrideName }} +data: + prometheus.yml: | + global: + scrape_interval: 1m + scrape_timeout: 10s + evaluation_interval: 1m + scrape_configs: + - job_name: httpServiceDiscovery + http_sd_configs: + - url: {{ printf "http://metering-svc.%s.svc%s:8000/v0/listScrapeTargets" .Release.Namespace $cluster_domain }} +{{- if or .Values.workerPodMetricSidecar.enabled .Values.kanisterPodMetricSidecar.enabled }} + - job_name: pushAggregator + honor_timestamps: true + metrics_path: /v0/push-metric-agg/metrics + static_configs: + - targets: + - {{ printf "metering-svc.%s.svc%s:8000" .Release.Namespace $cluster_domain }} +{{- end -}} +{{- if .Values.prometheus.scrapeCAdvisor }} + - job_name: 'kubernetes-cadvisor' + scheme: https + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + kubernetes_sd_configs: + - role: node + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_node_label_(.+) + - target_label: __address__ + replacement: kubernetes.default.svc:443 + - source_labels: [__meta_kubernetes_node_name] + regex: (.+) + target_label: __metrics_path__ + replacement: /api/v1/nodes/${1}/proxy/metrics/cadvisor +{{- end}} + - job_name: prometheus + metrics_path: {{ .Values.prometheus.server.baseURL }}metrics + static_configs: + - targets: + - "localhost:9090" + labels: + app: prometheus + component: server + - job_name: k10-pods + scheme: http + metrics_path: /metrics + kubernetes_sd_configs: + - role: pod + namespaces: + own_namespace: true + selectors: + - role: pod + label: "component=executor" + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_pod_container_port_number] + action: keep + regex: 8\d{3} +{{- if ne .Values.metering.mode "airgap" }} + - job_name: k10-grafana + scheme: http + metrics_path: /metrics + kubernetes_sd_configs: + - role: pod + namespaces: + own_namespace: true + selectors: + - role: pod + label: "component=grafana" + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_pod_container_port_number] + action: keep + regex: 3000 + metric_relabel_configs: + - source_labels: [__name__] + regex: grafana_http_request_duration_seconds_count + action: keep +{{- end}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/prometheus-scc.yaml b/charts/kasten/k10/7.0.1401/templates/prometheus-scc.yaml new file mode 100644 index 000000000..4d039ef00 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/prometheus-scc.yaml @@ -0,0 +1,41 @@ +{{- if .Values.scc.create }} +kind: SecurityContextConstraints +apiVersion: security.openshift.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Name }}-prometheus-server +allowPrivilegedContainer: false +allowHostNetwork: false +allowHostDirVolumePlugin: true +allowHostPorts: true +allowHostPID: false +allowHostIPC: false +readOnlyRootFilesystem: false +requiredDropCapabilities: +- CHOWN +- KILL +- MKNOD +- SETUID +- SETGID +defaultAddCapabilities: [] +allowedCapabilities: [] +priority: 0 +runAsUser: + type: MustRunAsNonRoot +seLinuxContext: + type: RunAsAny +fsGroup: + type: RunAsAny +supplementalGroups: + type: RunAsAny +volumes: +- configMap +- downwardAPI +- emptyDir +- persistentVolumeClaim +- projected +- secret +users: + - system:serviceaccount:{{.Release.Namespace}}:prometheus-server +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/prometheus-service.yaml b/charts/kasten/k10/7.0.1401/templates/prometheus-service.yaml new file mode 100644 index 000000000..a5a228171 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/prometheus-service.yaml @@ -0,0 +1,46 @@ +{{/* Template to generate service spec for v0 rest services */}} +{{- if .Values.prometheus.server.enabled -}} +{{- $postfix := default .Release.Name .Values.ingress.urlPath -}} +{{- $os_postfix := default .Release.Name .Values.route.path -}} +{{- $service_port := .Values.prometheus.server.service.servicePort -}} +apiVersion: v1 +kind: Service +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "k10.prometheus.service.name" . }}-exp + labels: +{{ include "helm.labels" $ | indent 4 }} + component: {{ include "k10.prometheus.service.name" . }} + run: {{ include "k10.prometheus.service.name" . }} + annotations: + getambassador.io/config: | + --- + apiVersion: getambassador.io/v3alpha1 + kind: Mapping + name: {{ include "k10.prometheus.service.name" . }}-mapping + {{- if .Values.prometheus.server.baseURL }} + rewrite: /{{ .Values.prometheus.server.baseURL | trimPrefix "/" | trimSuffix "/" }}/ + {{- else }} + rewrite: / + {{- end }} + {{- if .Values.route.enabled }} + prefix: /{{ $os_postfix | trimPrefix "/" | trimSuffix "/" }}/prometheus/ + {{- else }} + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/prometheus/ + {{- end }} + service: {{ include "k10.prometheus.service.name" . }}:{{ $service_port }} + timeout_ms: 15000 + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + +spec: + ports: + - name: http + protocol: TCP + port: {{ $service_port }} + targetPort: 9090 + selector: + app: {{ include "k10.prometheus.name" . }} + component: {{ .Values.prometheus.server.name }} + release: {{ .Release.Name }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/rbac.yaml b/charts/kasten/k10/7.0.1401/templates/rbac.yaml new file mode 100644 index 000000000..ec68013e9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/rbac.yaml @@ -0,0 +1,381 @@ +{{- $main := . -}} +{{- $apiDomain := include "apiDomain" . -}} + +{{- $actionsAPIs := splitList " " (include "k10.actionsAPIs" .) -}} +{{- $aggregatedAPIs := splitList " " (include "k10.aggregatedAPIs" .) -}} +{{- $appsAPIs := splitList " " (include "k10.appsAPIs" .) -}} +{{- $authAPIs := splitList " " (include "k10.authAPIs" .) -}} +{{- $configAPIs := splitList " " (include "k10.configAPIs" .) -}} +{{- $distAPIs := splitList " " (include "k10.distAPIs" .) -}} +{{- $reportingAPIs := splitList " " (include "k10.reportingAPIs" .) -}} + +{{- if .Values.rbac.create }} +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Namespace }}-{{ template "serviceAccountName" . }}-cluster-admin +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: cluster-admin +subjects: +- kind: ServiceAccount + name: {{ template "serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- if not ( eq (include "meteringServiceAccountName" .) (include "serviceAccountName" .) )}} +- kind: ServiceAccount + name: {{ template "meteringServiceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-admin +rules: +- apiGroups: +{{- range sortAlpha (concat $aggregatedAPIs $configAPIs $reportingAPIs) }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - "*" + verbs: + - "*" +- apiGroups: + - cr.kanister.io + resources: + - '*' + verbs: + - '*' +- apiGroups: + - "" + resources: + - namespaces + verbs: + - create + - get + - list +--- +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-ns-admin + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: + - "apps" + resources: + - deployments + verbs: + - get + - update + - watch + - list +- apiGroups: + - "" + resources: + - pods + verbs: + - get + - create + - delete + - list +- apiGroups: + - "apik10.kasten.io" + resources: + - k10s + verbs: + - list + - patch +- apiGroups: + - "" + resources: + - persistentvolumeclaims + verbs: + - get +- apiGroups: + - "" + resources: + - secrets + verbs: + - create + - delete + - get + - list + - update +- apiGroups: + - "" + resources: + - configmaps + verbs: + - create + - delete + - get + - list + - update +- apiGroups: + - "batch" + resources: + - jobs + verbs: + - get +- apiGroups: + - "" + resources: + - services + verbs: + - create + - get + - delete +- apiGroups: + - "networking.k8s.io" + resources: + - networkpolicies + verbs: + - get + - create + - list + - delete +- apiGroups: + - "" + resources: + - endpoints + verbs: + - list + - get +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-mc-admin +rules: +- apiGroups: +{{- range sortAlpha (concat $authAPIs $configAPIs $distAPIs) }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - "*" + verbs: + - "*" +- apiGroups: + - "" + resources: + - secrets + verbs: + - "*" +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-basic +rules: +- apiGroups: +{{- range sortAlpha $actionsAPIs }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - {{ include "k10.backupActions" $main}} + - {{ include "k10.backupActionsDetails" $main}} + - {{ include "k10.restoreActions" $main}} + - {{ include "k10.restoreActionsDetails" $main}} + - {{ include "k10.exportActions" $main}} + - {{ include "k10.exportActionsDetails" $main}} + - {{ include "k10.cancelActions" $main}} + - {{ include "k10.runActions" $main}} + - {{ include "k10.runActionsDetails" $main}} + verbs: + - "*" +- apiGroups: +{{- range sortAlpha $appsAPIs }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - {{ include "k10.restorePoints" $main}} + - {{ include "k10.restorePointsDetails" $main}} + - {{ include "k10.applications" $main}} + - {{ include "k10.applicationsDetails" $main}} + verbs: + - "*" +- apiGroups: + - "" + resources: + - namespaces + verbs: + - get +- apiGroups: +{{- range sortAlpha $configAPIs }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - {{ include "k10.policies" $main}} + verbs: + - "*" +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-config-view +rules: +- apiGroups: +{{- range sortAlpha $configAPIs }} + - {{ . }}.{{ $apiDomain }} +{{- end }} + resources: + - {{ include "k10.auditconfigs" $main}} + - {{ include "k10.profiles" $main}} + - {{ include "k10.policies" $main}} + - {{ include "k10.policypresets" $main}} + - {{ include "k10.transformsets" $main}} + - {{ include "k10.blueprintbindings" $main}} + - {{ include "k10.storagesecuritycontexts" $main}} + - {{ include "k10.storagesecuritycontextbindings" $main}} + verbs: + - get + - list +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Namespace }}-{{ template "serviceAccountName" . }}-admin +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-admin +subjects: +- apiGroup: rbac.authorization.k8s.io + kind: Group + name: k10:admins +{{- range .Values.auth.k10AdminUsers }} +- apiGroup: rbac.authorization.k8s.io + kind: User + name: {{ . }} +{{- end }} +{{- range default .Values.auth.groupAllowList .Values.auth.k10AdminGroups }} +- apiGroup: rbac.authorization.k8s.io + kind: Group + name: {{ . }} +{{- end }} +--- +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Namespace }}-{{ template "serviceAccountName" . }}-ns-admin + namespace: {{ .Release.Namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ .Release.Name }}-ns-admin +subjects: +- apiGroup: rbac.authorization.k8s.io + kind: Group + name: k10:admins +{{- range .Values.auth.k10AdminUsers }} +- apiGroup: rbac.authorization.k8s.io + kind: User + name: {{ . }} +{{- end }} +{{- range default .Values.auth.groupAllowList .Values.auth.k10AdminGroups }} +- apiGroup: rbac.authorization.k8s.io + kind: Group + name: {{ . }} +{{- end }} +--- +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Namespace }}-{{ template "serviceAccountName" . }}-mc-admin +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-mc-admin +subjects: + - apiGroup: rbac.authorization.k8s.io + kind: Group + name: k10:admins +{{- range .Values.auth.k10AdminUsers }} + - apiGroup: rbac.authorization.k8s.io + kind: User + name: {{ . }} +{{- end }} +{{- range default .Values.auth.groupAllowList .Values.auth.k10AdminGroups }} + - apiGroup: rbac.authorization.k8s.io + kind: Group + name: {{ . }} +{{- end }} +{{- end }} +{{- if and .Values.rbac.create (not .Values.prometheus.rbac.create) }} +--- +kind: Role +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} +{{ include "k10.defaultRBACLabels" . | indent 4 }} + name: {{ .Release.Name }}-prometheus-server + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: + - "" + resources: + - nodes + - nodes/proxy + - nodes/metrics + - services + - endpoints + - pods + - ingresses + - configmaps + verbs: + - get + - list + - watch +- apiGroups: + - extensions + - networking.k8s.io + resources: + - ingresses/status + - ingresses + verbs: + - get + - list + - watch +--- +kind: RoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ .Release.Namespace }}-{{ template "serviceAccountName" . }}-prometheus-server + namespace: {{ .Release.Namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ .Release.Name }}-prometheus-server +subjects: + - kind: ServiceAccount + name: prometheus-server + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/route.yaml b/charts/kasten/k10/7.0.1401/templates/route.yaml new file mode 100644 index 000000000..1ecd244be --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/route.yaml @@ -0,0 +1,36 @@ +{{- $route := .Values.route -}} +{{- if $route.enabled -}} +{{ include "authEnabled.check" . }} +apiVersion: route.openshift.io/v1 +kind: Route +metadata: + name: {{ .Release.Name }}-route + {{- with $route.annotations }} + namespace: {{ .Release.Namespace }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: +{{ include "helm.labels" . | indent 4 }} + {{- with $route.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + host: {{ $route.host }} + path: /{{ default .Release.Name $route.path | trimPrefix "/" | trimSuffix "/" }}/ + port: + targetPort: http + to: + kind: Service + name: gateway + weight: 100 + {{- if $route.tls.enabled }} + tls: + {{- if $route.tls.insecureEdgeTerminationPolicy }} + insecureEdgeTerminationPolicy: {{ $route.tls.insecureEdgeTerminationPolicy }} + {{- end }} + {{- if $route.tls.termination }} + termination: {{ $route.tls.termination }} + {{- end }} + {{- end }} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/secrets.yaml b/charts/kasten/k10/7.0.1401/templates/secrets.yaml new file mode 100644 index 000000000..0a040e2c0 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/secrets.yaml @@ -0,0 +1,257 @@ +{{- include "enforce.singlecloudcreds" . -}} +{{- include "enforce.singleazurecreds" . -}} +{{- include "check.validateImagePullSecrets" . -}} +{{- if and (eq (include "check.awscreds" . ) "true") (not (eq (include "check.awsSecretName" . ) "true")) }} +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: aws-creds +type: Opaque +data: + aws_access_key_id: {{ required "secrets.awsAccessKeyId field is required!" .Values.secrets.awsAccessKeyId | b64enc | quote }} + aws_secret_access_key: {{ required "secrets.awsSecretAccessKey field is required!" .Values.secrets.awsSecretAccessKey | b64enc | quote }} +{{- if .Values.secrets.awsIamRole }} + role: {{ .Values.secrets.awsIamRole | trim | b64enc | quote }} +{{- end }} +{{- end }} +{{- if or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: k10-ecr +type: kubernetes.io/dockerconfigjson +data: + .dockerconfigjson: {{ or .Values.secrets.dockerConfig ( .Values.secrets.dockerConfigPath | b64enc ) }} +{{- end }} +{{- if and (eq (include "check.googlecreds" .) "true") ( not (eq (include "check.googleCredsSecret" .) "true")) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: google-secret +type: Opaque +data: + kasten-gke-sa.json: {{ .Values.secrets.googleApiKey }} +{{- if eq (include "check.googleproject" .) "true" }} + kasten-gke-project: {{ .Values.secrets.googleProjectId | b64enc }} +{{- end }} +{{- end }} +{{- if eq (include "check.azurecreds" .) "true" }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: azure-creds +type: Opaque +data: + {{- if not (eq (include "check.azuresecret" .) "true" ) }} + {{- if or (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureClientSecretCreds" .) "true") }} + azure_client_id: {{ required "secrets.azureClientId field is required!" .Values.secrets.azureClientId | b64enc | quote }} + {{- end }} + {{- if eq (include "check.azureClientSecretCreds" .) "true" }} + azure_tenant_id: {{ required "secrets.azureTenantId field is required!" .Values.secrets.azureTenantId | b64enc | quote }} + azure_client_secret: {{ required "secrets.azureClientSecret field is required!" .Values.secrets.azureClientSecret | b64enc | quote }} + {{- end }} + {{- end }} + azure_resource_group: {{ default "" .Values.secrets.azureResourceGroup | b64enc | quote }} + azure_subscription_id: {{ default "" .Values.secrets.azureSubscriptionID | b64enc | quote }} + azure_resource_manager_endpoint: {{ default "" .Values.secrets.azureResourceMgrEndpoint | b64enc | quote }} + entra_id_endpoint: {{ default "" (default .Values.secrets.azureADEndpoint .Values.secrets.microsoftEntraIDEndpoint) | b64enc | quote }} + entra_id_resource_id: {{ default "" (default .Values.secrets.azureADResourceID .Values.secrets.microsoftEntraIDResourceID) | b64enc | quote }} + azure_cloud_env_id: {{ default "" .Values.secrets.azureCloudEnvID | b64enc | quote }} +{{- end }} +{{- if and (eq (include "check.vspherecreds" .) "true") (not (eq (include "check.vsphereClientSecret" . ) "true")) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + namespace: {{ .Release.Namespace }} + name: vsphere-creds +type: Opaque +data: + vsphere_endpoint: {{ required "secrets.vsphereEndpoint field is required!" .Values.secrets.vsphereEndpoint | b64enc | quote }} + vsphere_username: {{ required "secrets.vsphereUsername field is required!" .Values.secrets.vsphereUsername | b64enc | quote }} + vsphere_password: {{ required "secrets.vspherePassword field is required!" .Values.secrets.vspherePassword | b64enc | quote }} +{{- end }} +{{- if and (eq (include "basicauth.check" .) "true") (not .Values.auth.basicAuth.secretName) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-basic-auth + namespace: {{ .Release.Namespace }} +data: + auth: {{ required "auth.basicAuth.htpasswd field is required!" .Values.auth.basicAuth.htpasswd | b64enc | quote}} +type: Opaque +{{- end }} +{{- if .Values.auth.tokenAuth.enabled }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-token-auth + namespace: {{ .Release.Namespace }} +data: + auth: {{ "true" | b64enc | quote}} +type: Opaque +{{- end }} +{{- if and .Values.auth.oidcAuth.enabled (not .Values.auth.oidcAuth.secretName) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ include "k10.oidcSecretName" .}} + namespace: {{ .Release.Namespace }} +data: + provider-url: {{ required "auth.oidcAuth.providerURL field is required!" .Values.auth.oidcAuth.providerURL | b64enc | quote }} + redirect-url: {{ required "auth.oidcAuth.redirectURL field is required!" .Values.auth.oidcAuth.redirectURL | b64enc | quote }} +{{- if not .Values.auth.oidcAuth.clientSecretName }} + client-id: {{ required "auth.oidcAuth.clientID field is required!" .Values.auth.oidcAuth.clientID | b64enc | quote }} + client-secret: {{ required "auth.oidcAuth.clientSecret field is required!" .Values.auth.oidcAuth.clientSecret | b64enc | quote }} +{{- end }} + scopes: {{ required "auth.oidcAuth.scopes field is required!" .Values.auth.oidcAuth.scopes | b64enc | quote }} + prompt: {{ default "select_account" .Values.auth.oidcAuth.prompt | b64enc | quote }} + usernameClaim: {{ default "sub" .Values.auth.oidcAuth.usernameClaim | b64enc | quote }} + usernamePrefix: {{ default "" .Values.auth.oidcAuth.usernamePrefix | b64enc | quote }} + groupClaim: {{ default "" .Values.auth.oidcAuth.groupClaim | b64enc | quote }} + groupPrefix: {{ default "" .Values.auth.oidcAuth.groupPrefix | b64enc | quote }} + sessionDuration: {{ default "1h" .Values.auth.oidcAuth.sessionDuration | b64enc | quote }} +{{- if .Values.auth.oidcAuth.refreshTokenSupport }} + refreshTokenSupport: {{ "true" | b64enc | quote }} +{{- else }} + refreshTokenSupport: {{ "false" | b64enc | quote }} +{{ end }} +stringData: + groupAllowList: |- +{{- range $.Values.auth.groupAllowList }} + {{ . -}} +{{ end }} + logout-url: {{ default "" .Values.auth.oidcAuth.logoutURL | b64enc | quote }} +type: Opaque +{{- end }} +{{- if and (.Values.auth.openshift.enabled) (and (not .Values.auth.openshift.clientSecretName) (not .Values.auth.openshift.clientSecret)) }} +--- +apiVersion: v1 +kind: Secret +type: kubernetes.io/service-account-token +metadata: + name: {{ include "get.openshiftServiceAccountSecretName" . }} + annotations: + kubernetes.io/service-account.name: {{ include "get.openshiftServiceAccountName" . | quote }} +{{- end }} +{{- if and (.Values.auth.openshift.enabled) (not .Values.auth.openshift.secretName) }} +{{ $dashboardURL := required "auth.openshift.dashboardURL field is required!" .Values.auth.openshift.dashboardURL }} +{{ $redirectURL := trimSuffix "/" (trimSuffix (default .Release.Name .Values.ingress.urlPath) (trimSuffix "/" $dashboardURL)) | b64enc | quote }} +{{- if .Values.route.enabled }} +{{ $redirectURL := trimSuffix "/" (trimSuffix (default .Release.Name .Values.route.path) (trimSuffix "/" $dashboardURL)) | b64enc | quote }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ include "k10.oidcSecretName" .}} + namespace: {{ .Release.Namespace }} +data: + provider-url: {{ printf "%s/dex" (trimSuffix "/" $dashboardURL) | b64enc | quote }} + redirect-url: {{ $redirectURL }} + client-id: {{ (printf "kasten") | b64enc | quote }} + client-secret: {{ (printf "kastensecret") | b64enc | quote }} + scopes: {{ (printf "groups profile email") | b64enc | quote }} + prompt: {{ (printf "select_account") | b64enc | quote }} + usernameClaim: {{ default "email" .Values.auth.openshift.usernameClaim | b64enc | quote }} + usernamePrefix: {{ default "" .Values.auth.openshift.usernamePrefix | b64enc | quote }} + groupClaim: {{ default "groups" .Values.auth.openshift.groupClaim | b64enc | quote }} + groupPrefix: {{ default "" .Values.auth.openshift.groupPrefix | b64enc | quote }} +stringData: + groupAllowList: |- +{{- range $.Values.auth.groupAllowList }} + {{ . -}} +{{ end }} +type: Opaque +{{- end }} +{{- if and .Values.auth.ldap.enabled (not .Values.auth.ldap.secretName) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ include "k10.oidcSecretName" .}} + namespace: {{ .Release.Namespace }} +data: + provider-url: {{ required "auth.ldap.dashboardURL field is required!" (printf "%s/dex" (trimSuffix "/" .Values.auth.ldap.dashboardURL)) | b64enc | quote }} + {{- if .Values.route.enabled }} + redirect-url: {{ required "auth.ldap.dashboardURL field is required!" (trimSuffix "/" (trimSuffix (default .Release.Name .Values.route.path) (trimSuffix "/" .Values.auth.ldap.dashboardURL))) | b64enc | quote }} + {{- else }} + redirect-url: {{ required "auth.ldap.dashboardURL field is required!" (trimSuffix "/" (trimSuffix (default .Release.Name .Values.ingress.urlPath) (trimSuffix "/" .Values.auth.ldap.dashboardURL))) | b64enc | quote }} + {{- end }} + client-id: {{ (printf "kasten") | b64enc | quote }} + client-secret: {{ (printf "kastensecret") | b64enc | quote }} + scopes: {{ (printf "groups profile email") | b64enc | quote }} + prompt: {{ (printf "select_account") | b64enc | quote }} + usernameClaim: {{ default "email" .Values.auth.ldap.usernameClaim | b64enc | quote }} + usernamePrefix: {{ default "" .Values.auth.ldap.usernamePrefix | b64enc | quote }} + groupClaim: {{ default "groups" .Values.auth.ldap.groupClaim | b64enc | quote }} + groupPrefix: {{ default "" .Values.auth.ldap.groupPrefix | b64enc | quote }} +stringData: + groupAllowList: |- +{{- range $.Values.auth.groupAllowList }} + {{ . -}} +{{ end }} +type: Opaque +{{- end }} +{{- if and .Values.auth.ldap.enabled (not .Values.auth.ldap.bindPWSecretName) }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-dex + namespace: {{ .Release.Namespace }} +data: + bindPW: {{ required "auth.ldap.bindPW field is required!" .Values.auth.ldap.bindPW | b64enc | quote }} +type: Opaque +{{- end }} +{{- if eq (include "check.primaryKey" . ) "true" }} +--- +apiVersion: v1 +kind: Secret +metadata: + labels: +{{ include "helm.labels" . | indent 4 }} + name: k10-encryption-primary-key + namespace: {{ .Release.Namespace }} +data: + {{- if .Values.encryption.primaryKey.awsCmkKeyId }} + awscmkkeyid: {{ default "" .Values.encryption.primaryKey.awsCmkKeyId | trim | b64enc | quote }} + {{- end }} + {{- if .Values.encryption.primaryKey.vaultTransitKeyName }} + vaulttransitkeyname: {{ default "" .Values.encryption.primaryKey.vaultTransitKeyName | trim | b64enc | quote }} + vaulttransitpath: {{ default "transit" .Values.encryption.primaryKey.vaultTransitPath | trim | b64enc | quote }} + {{- end }} +type: Opaque +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/secure_deployment.tpl b/charts/kasten/k10/7.0.1401/templates/secure_deployment.tpl new file mode 100644 index 000000000..55c537eb9 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/secure_deployment.tpl @@ -0,0 +1,17 @@ +{{/* +This file is used to fail the helm deployment if certain values are set which are +not compatible with a secure deployment. + +A secure deployment is defined as one of the following: +- Iron Bank +- FIPS +*/}} + +{{/* Iron Bank */}} +{{- include "k10.fail.ironbankGrafana" . -}} +{{- include "k10.fail.ironbankPrometheus" . -}} +{{- include "k10.fail.ironbankRHMarketplace" . -}} + +{{/* FIPS */}} +{{- include "k10.fail.fipsGrafana" . -}} +{{- include "k10.fail.fipsPrometheus" . -}} diff --git a/charts/kasten/k10/7.0.1401/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1401/templates/serviceaccount.yaml new file mode 100644 index 000000000..d24d91e16 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/serviceaccount.yaml @@ -0,0 +1,48 @@ +{{- if and .Values.serviceAccount.create ( not .Values.metering.awsMarketplace ) }} +kind: ServiceAccount +apiVersion: v1 +metadata: +{{- if .Values.secrets.awsIamRole }} + annotations: + eks.amazonaws.com/role-arn: {{ .Values.secrets.awsIamRole }} +{{- end }} +{{- if eq (include "check.azureFederatedIdentity" .) "true" }} + annotations: + azure.workload.identity/client-id: {{ .Values.secrets.azureClientId | quote }} +{{- end }} + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ template "serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +{{- if and (not ( eq (include "meteringServiceAccountName" .) (include "serviceAccountName" .))) ( not .Values.metering.awsManagedLicense ) .Values.metering.serviceAccount.create }} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: +{{- if .Values.metering.awsMarketPlaceIamRole }} + annotations: + eks.amazonaws.com/role-arn: {{ .Values.metering.awsMarketPlaceIamRole }} +{{- end }} + labels: +{{ include "helm.labels" . | indent 4 }} + name: {{ template "meteringServiceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +{{- if and (.Values.auth.openshift.enabled) (not .Values.auth.openshift.serviceAccount) }} +{{- if or (.Values.auth.openshift.clientSecret) (.Values.auth.openshift.clientSecretName) }} + {{ fail "auth.openshift.serviceAccount is required when auth.openshift.clientSecret or auth.openshift.clientSecretName is used "}} +{{- end }} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: {{ include "k10.dexServiceAccountName" . }} + namespace: {{ .Release.Namespace }} + annotations: + {{- $dashboardURL := (trimSuffix "/" (required "auth.openshift.dashboardURL field is required" .Values.auth.openshift.dashboardURL)) -}} + {{- if (not (hasSuffix .Release.Name $dashboardURL)) }} + {{ fail "auth.openshift.dashboardURL should end with the K10's release name" }} + {{- end }} + serviceaccounts.openshift.io/oauth-redirecturi.dex: {{ printf "%s/dex/callback" $dashboardURL }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/v0services.yaml b/charts/kasten/k10/7.0.1401/templates/v0services.yaml new file mode 100644 index 000000000..8a744cfe3 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/v0services.yaml @@ -0,0 +1,200 @@ +{{/* Template to generate service spec for v0 rest services */}} +{{- $container_port := .Values.service.internalPort -}} +{{- $service_port := .Values.service.externalPort -}} +{{- $aggregated_api_port := .Values.service.aggregatedApiPort -}} +{{- $postfix := default .Release.Name .Values.ingress.urlPath -}} +{{- $colocated_services := include "get.enabledColocatedServices" . | fromYaml -}} +{{- $exposed_services := include "get.enabledExposedServices" . | splitList " " -}} +{{- $os_postfix := default .Release.Name .Values.route.path -}} +{{- $main_context := . -}} +{{ $service_list := append (include "get.enabledRestServices" . | splitList " ") "frontend" }} +{{- range $service_list }} + {{- $exposed_service := (has . $exposed_services) }} + {{- $mc_exposed_service := (eq . "controllermanager") }} + {{ if not (hasKey $colocated_services . ) }} +apiVersion: v1 +kind: Service +metadata: + namespace: {{ $.Release.Namespace }} + name: {{ . }}-svc + labels: +{{ include "helm.labels" $ | indent 4 }} + component: {{ . }} + run: {{ . }}-svc +{{- if not (include "k10.capability.gateway" $) }} +{{- if or $exposed_service (eq . "frontend") $mc_exposed_service }} + annotations: + getambassador.io/config: | + {{- if or $exposed_service (eq . "frontend") }} + --- + apiVersion: getambassador.io/v3alpha1 + kind: Mapping + name: {{ . }}-mapping + {{- if $.Values.route.enabled }} + {{- if eq . "frontend" }} + prefix: /{{ $os_postfix | trimPrefix "/" | trimSuffix "/" }}/ + {{- else }} + prefix: /{{ $os_postfix | trimPrefix "/" | trimSuffix "/" }}/{{ . }}-svc/ + {{- end }} + {{- else }} + {{- if eq . "frontend" }} + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/ + {{- else }} + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/{{ . }}-svc/ + {{- end }} + {{- end }} + rewrite: / + service: {{ . }}-svc.{{ $.Release.Namespace }}:{{ $service_port }} + timeout_ms: 30000 + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + {{- end }} + {{- $colocatedList := include "get.enabledColocatedSvcList" $main_context | fromYaml }} + {{- range $skip, $secondary := index $colocatedList . }} + {{- $colocConfig := index (include "get.enabledColocatedServices" $main_context | fromYaml) $secondary }} + {{- if (has $secondary $exposed_services) }} + --- + apiVersion: getambassador.io/v3alpha1 + kind: Mapping + name: {{ $secondary }}-mapping + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/{{ $secondary }}-svc/ + rewrite: / + service: {{ $colocConfig.primary }}-svc.{{ $.Release.Namespace }}:{{ $colocConfig.port }} + timeout_ms: 30000 + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + {{- end }} + {{- end }} + {{- if $mc_exposed_service }} + --- + apiVersion: getambassador.io/v3alpha1 + kind: Mapping + name: {{ . }}-mc-mapping + {{- if $.Values.route.enabled }} + prefix: /{{ $os_postfix | trimPrefix "/" | trimSuffix "/" }}/mc/ + {{- else }} + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/mc/ + {{- end }} + rewrite: / + service: {{ . }}-svc.{{ $.Release.Namespace }}:{{ include "k10.mcExternalPort" nil }} + timeout_ms: 30000 + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] + {{- end }} +{{- end }} +{{- end }} +spec: + ports: + - name: http + protocol: TCP + port: {{ $service_port }} + targetPort: {{ $container_port }} + {{- if and (eq . "controllermanager") ($.Values.injectKanisterSidecar.enabled) }} + - name: https + protocol: TCP + port: 443 + targetPort: {{ $.Values.injectKanisterSidecar.webhookServer.port }} + {{- end }} +{{- $colocatedList := include "get.enabledColocatedSvcList" $main_context | fromYaml }} +{{- range $skip, $secondary := index $colocatedList . }} + {{- $colocConfig := index (include "get.enabledColocatedServices" $main_context | fromYaml) $secondary }} + - name: {{ $secondary }} + protocol: TCP + port: {{ $colocConfig.port }} + targetPort: {{ $colocConfig.port }} +{{- end }} +{{- if eq . "logging" }} + - name: logging + protocol: TCP + port: 24224 + targetPort: 24224 + - name: logging-metrics + protocol: TCP + port: 24225 + targetPort: 24225 +{{- end }} +{{- if eq . "controllermanager" }} + - name: mc-http + protocol: TCP + port: {{ include "k10.mcExternalPort" nil }} + targetPort: {{ include "k10.mcExternalPort" nil }} +{{- end }} + selector: + run: {{ . }}-svc +--- + {{ end }}{{/* if not (hasKey $colocated_services $k10_service ) */}} +{{ end -}}{{/* range append (include "get.enabledRestServices" . | splitList " ") "frontend" */}} +{{- range append (include "get.enabledServices" . | splitList " ") "kanister" }} +{{- if eq . "gateway" -}}{{- continue -}}{{- end -}} +apiVersion: v1 +kind: Service +metadata: + namespace: {{ $.Release.Namespace }} + name: {{ . }}-svc + labels: +{{ include "helm.labels" $ | indent 4 }} + component: {{ . }} + run: {{ . }}-svc +spec: + ports: + {{- if eq . "aggregatedapis" }} + - name: http + port: 443 + protocol: TCP + targetPort: {{ $aggregated_api_port }} + {{- else }} + - name: http + protocol: TCP + port: {{ $service_port }} + targetPort: {{ $container_port }} + {{- end }} +{{- $colocatedList := include "get.enabledColocatedSvcList" $main_context | fromYaml }} +{{- range $skip, $secondary := index $colocatedList . }} + {{- $colocConfig := index (include "get.enabledColocatedServices" . | fromYaml) $secondary }} + - name: {{ $secondary }} + protocol: TCP + port: {{ $colocConfig.port }} + targetPort: {{ $colocConfig.port }} +{{- end }} + selector: + run: {{ . }}-svc +--- +{{ end -}} +{{- if eq (include "check.dexAuth" .) "true" }} +apiVersion: v1 +kind: Service +metadata: +{{- if not (include "k10.capability.gateway" $) }} + annotations: + getambassador.io/config: | + --- + apiVersion: getambassador.io/v3alpha1 + kind: Mapping + name: dex-mapping + {{- if $.Values.route.enabled }} + prefix: /{{ $os_postfix | trimPrefix "/" | trimSuffix "/" }}/dex/ + {{- else }} + prefix: /{{ $postfix | trimPrefix "/" | trimSuffix "/" }}/dex/ + {{- end }} + rewrite: "" + service: dex.{{ $.Release.Namespace }}:8000 + timeout_ms: 30000 + hostname: "*" + ambassador_id: [ {{ include "k10.ambassadorId" . }} ] +{{- end }} + name: dex + namespace: {{ $.Release.Namespace }} + labels: +{{ include "helm.labels" $ | indent 4 }} + component: dex + run: auth-svc +spec: + ports: + - name: http + port: {{ $service_port }} + protocol: TCP + targetPort: 8080 + selector: + run: auth-svc + type: ClusterIP +{{ end -}} diff --git a/charts/kasten/k10/7.0.1401/templates/workloadIdentityFederation.tpl b/charts/kasten/k10/7.0.1401/templates/workloadIdentityFederation.tpl new file mode 100644 index 000000000..75296e98b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/workloadIdentityFederation.tpl @@ -0,0 +1,10 @@ +{{/* +This file is used to fail the helm deployment if Workload Identity settings are not +compatible. +*/}} +{{- include "validate.gwif.idp.type" . -}} +{{- include "validate.gwif.idp.aud" . -}} + + + + diff --git a/charts/kasten/k10/7.0.1401/templates/{values}/grafana/values/grafana_values.tpl b/charts/kasten/k10/7.0.1401/templates/{values}/grafana/values/grafana_values.tpl new file mode 100644 index 000000000..3203a6c62 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/{values}/grafana/values/grafana_values.tpl @@ -0,0 +1,278 @@ +{{/* + With some of K10's features being provided by external Helm charts, those Helm + charts need to be configured to work with K10. + + Unfortunately, some of the values needed to configure the subcharts aren't + accessible to the subcharts (only global.* and chart_name.* are accessible). + + This means the values need to be duplicated, making the configuration of K10 + quite cumbersome for users (the same setting has to be provided in multiple + places, making it easy to misconfigure one thing or another). + + Alternatively, the subchart's templates could be customized to read global.* + values instead. However, this means upgrading the subchart is quite burdensome + since the customizations have to be re-applied to the upgraded chart. This is + even less tenable with the frequency with which chart updates are needed. + + With this in mind, this template was specially crafted to be able to read K10 + values and update the values that will be passed to the subchart. + + --- + + To accomplish this, Helm's template parsing and rendering order is exploited. + + Helm allows parent charts to override templates in subcharts. This is done by + parsing templates with lower precedence first (templates that are more deeply + nested than others). This allows templates with higher precedence to redefine + templates with lower precedence. + + Helm also renders templates in this same order. This template exploits this + ordering in order to set subchart values before the subchart's templates are + rendered, having the same effect as the user setting the values. + + WARNING: The name and directory structure of this template was carefully + selected to ensure that it is rendered before other templates! +*/}} + +{{- if .Values.grafana.enabled }} +{{- $grafana_prefix := printf "%s/grafana/" (include "k10.prefixPath" $) -}} +{{- $grafana_scoped_values := (dict "Chart" (dict "Name" "grafana") "Release" .Release "Values" .Values.grafana) -}} + +{{- /*** GRAFANA LABELS ***/ -}} +{{- /* Merge global pod labels with any grafana-specific labels, where the latter is of highest priority */ -}} +{{- $podLabels := merge (dict) (dict "component" "grafana") (.Values.grafana.podLabels | default dict) (.Values.global.podLabels) -}} + +{{- /* Merge global pod annotations with any grafana-specific annotations, where the latter is of highest priority */ -}} +{{- $podAnnotations := merge (dict) (.Values.grafana.podAnnotations | default dict) (.Values.global.podAnnotations) -}} +{{- if .Values.scc.create -}} + {{- $podAnnotations = merge (dict "openshift.io/required-scc" (printf "%s-grafana" .Release.Name)) $podAnnotations -}} +{{- end -}} +{{- $_ := mergeOverwrite .Values.grafana + (dict + "extraLabels" (dict + "app.kubernetes.io/name" (include "grafana.name" $grafana_scoped_values) + "app.kubernetes.io/instance" .Release.Name + "component" "grafana" + ) + "podLabels" $podLabels + "podAnnotations" $podAnnotations + ) +-}} + +{{- /*** GRAFANA SERVER CONFIGURATION ***/ -}} +{{- $_ := mergeOverwrite (index .Values.grafana "grafana.ini") + (dict + "auth" (dict + "disable_login_form" true + "disable_signout_menu" true + ) + "auth.basic" (dict + "enabled" false + ) + "auth.anonymous" (dict + "enabled" true + ) + "server" (dict + "root_url" $grafana_prefix + ) + ) +-}} +{{- $authAnonymous := index .Values.grafana "grafana.ini" "auth.anonymous" -}} +{{- $_ := set $authAnonymous "org_name" ($authAnonymous.org_name | default "Main Org.") -}} +{{- $_ := set $authAnonymous "org_role" ($authAnonymous.org_role | default "Admin") -}} + +{{- /*** GRAFANA DEPLOYMENT STRATEGY ***/ -}} +{{- $_ := set .Values.grafana.deploymentStrategy "type" "Recreate" -}} + +{{- /*** GRAFANA NETWORKING POLICY ***/ -}} +{{- $_ := set .Values.grafana.networkPolicy "enabled" true -}} + +{{- /*** GRAFANA TEST FRAMEWORK ***/ -}} +{{- $_ := set .Values.grafana.testFramework "enabled" false -}} + +{{- /*** GRAFANA RBAC ***/ -}} +{{- $_ := set .Values.grafana.rbac "namespaced" true -}} + +{{- /*** K10 PROMETHEUS DATASOURCE ***/ -}} +{{- $_ := set .Values.grafana.datasources + "datasources.yaml" (dict + "apiVersion" 1 + "datasources" (list + (dict + "access" "proxy" + "editable" false + "isDefault" true + "name" "Prometheus" + "type" "prometheus" + "url" (printf "http://%s-exp%s" (include "k10.prometheus.service.name" $) .Values.prometheus.server.baseURL) + "jsonData" (dict + "timeInterval" "1m" + ) + ) + ) + ) +-}} + +{{- /*** K10 DASHBOARD ***/ -}} +{{- $_ := set .Values.grafana.dashboards + "default" (dict + "default" (dict + "json" (.Files.Get "grafana/dashboards/default/default.json") + ) + ) +-}} + +{{- $_ := mergeOverwrite (index .Values.grafana "grafana.ini") + (dict + "dashboards" (dict + "default_home_dashboard_path" "/var/lib/grafana/dashboards/default/default.json" + ) + ) +-}} + +{{- $_ := set .Values.grafana.dashboardProviders + "dashboardproviders.yaml" (dict + "apiVersion" 1 + "providers" (list + (dict + "name" "default" + "orgId" 1 + "folder" "" + "type" "file" + "disableDeletion" true + "editable" false + "options" (dict + "path" "/var/lib/grafana/dashboards" + ) + ) + ) + ) +-}} + +{{- /*** K10 PERSISTENCE *** + - global.persistence.enabled + - global.persistence.accessMode + - global.persistence.storageClass + - global.persistence.grafana.size + - global.persistence.size +*/ -}} +{{- if .Values.global.persistence.enabled -}} + {{ $grafana_storage_class := dict }} + {{- if eq .Values.global.persistence.storageClass "-" -}} + {{ $grafana_storage_class = (dict "storageClassName" "") }} + {{- else if .Values.global.persistence.storageClass -}} + {{ $grafana_storage_class = (dict "storageClassName" .Values.global.persistence.storageClass) }} + {{- end -}} + + {{- $_ := mergeOverwrite .Values.grafana.persistence + $grafana_storage_class + (dict + "enabled" true + "accessModes" (list .Values.global.persistence.accessMode) + "size" (.Values.global.persistence.grafana.size | default .Values.global.persistence.size) + ) + -}} +{{- end -}} + +{{- /*** K10 IMAGE PULL SECRETS *** + - secrets.dockerConfig + - secrets.dockerConfigPath + - global.imagePullSecret +*/ -}} +{{- $image_pull_secrets := list -}} +{{- if .Values.global.imagePullSecret -}} + {{- $image_pull_secrets = append $image_pull_secrets .Values.global.imagePullSecret -}} +{{- end -}} +{{- if (or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath) -}} + {{ $image_pull_secrets = append $image_pull_secrets "k10-ecr" -}} +{{- end -}} +{{- $image_pull_secrets = $image_pull_secrets | compact | uniq -}} + +{{- if $image_pull_secrets -}} + {{- $image_pull_secrets = concat (.Values.grafana.image.pullSecrets | default list) $image_pull_secrets -}} + {{- $_ := set .Values.grafana.image "pullSecrets" $image_pull_secrets -}} +{{- end -}} + +{{- /*** K10 GRAFANA IMAGE *** + - global.airgapped.repository + - global.image.registry + - global.image.tag + - global.images.grafana +*/ -}} +{{- $grafana_image := (dict + "registry" (.Values.global.airgapped.repository | default .Values.global.image.registry) + "repository" "grafana" + "tag" (include "get.k10ImageTag" $) +) -}} +{{- if .Values.global.images.grafana -}} + {{- $grafana_image_args := (dict "image" .Values.global.images.grafana "path" "global.images.grafana") -}} + {{- $grafana_image = (include "k10.splitImage" $grafana_image_args) | fromJson -}} +{{- end -}} + +{{- if .Values.global.azMarketPlace -}} + {{- $grafana_image = ( dict + "registry" .Values.global.azure.images.grafana.registry + "repository" .Values.global.azure.images.grafana.image + "tag" .Values.global.azure.images.grafana.tag + ) + -}} +{{- end -}} + +{{- $_ := set .Values.grafana.image "registry" $grafana_image.registry -}} +{{- $_ := set .Values.grafana.image "repository" $grafana_image.repository -}} +{{- $_ := set .Values.grafana.image "tag" $grafana_image.tag -}} +{{- $_ := set .Values.grafana.image "sha" $grafana_image.sha -}} + +{{- /*** K10 INIT IMAGE *** + - global.airgapped.repository + - global.image.registry + - global.image.tag + - global.images.init +*/ -}} +{{- $init_image := (dict + "registry" (.Values.global.airgapped.repository | default .Values.global.image.registry) + "repository" "init" + "tag" (include "get.k10ImageTag" $) +) -}} + +{{- if .Values.global.images.init -}} + {{- $init_image_args := (dict "image" .Values.global.images.init "path" "global.images.init") -}} + {{- $init_image = (include "k10.splitImage" $init_image_args) | fromJson -}} +{{- end -}} + +{{- if .Values.global.azMarketPlace -}} + {{- $init_image = ( dict + "registry" .Values.global.azure.images.init.registry + "repository" .Values.global.azure.images.init.image + "tag" .Values.global.azure.images.init.tag + ) + -}} +{{- end -}} + +{{- $_ := set .Values.grafana.downloadDashboardsImage "registry" $init_image.registry -}} +{{- $_ := set .Values.grafana.downloadDashboardsImage "repository" $init_image.repository -}} +{{- $_ := set .Values.grafana.downloadDashboardsImage "tag" $init_image.tag -}} +{{- $_ := set .Values.grafana.downloadDashboardsImage "sha" $init_image.sha -}} + +{{- $_ := set .Values.grafana.initChownData.image "registry" $init_image.registry -}} +{{- $_ := set .Values.grafana.initChownData.image "repository" $init_image.repository -}} +{{- $_ := set .Values.grafana.initChownData.image "tag" $init_image.tag -}} +{{- $_ := set .Values.grafana.initChownData.image "sha" $init_image.sha -}} + +{{- /*** K10 SERVICE ***/ -}} +{{- $_ := set .Values.grafana.service.annotations + "getambassador.io/config" (dict + "apiVersion" "getambassador.io/v3alpha1" + "kind" "Mapping" + "name" "grafana-server-mapping" + "prefix" $grafana_prefix + "rewrite" "/" + "service" (printf "%s-grafana:%0.f" .Release.Name .Values.grafana.service.port) + "timeout_ms" 15000 + "hostname" "*" + "ambassador_id" (list + (include "k10.ambassadorId" nil | replace "\"" "") + ) + | toYaml) +-}} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl b/charts/kasten/k10/7.0.1401/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl new file mode 100644 index 000000000..fbf1f2ef6 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl @@ -0,0 +1,186 @@ +{{/* + With some of K10's features being provided by external Helm charts, those Helm + charts need to be configured to work with K10. + + Unfortunately, some of the values needed to configure the subcharts aren't + accessible to the subcharts (only global.* and chart_name.* are accessible). + + This means the values need to be duplicated, making the configuration of K10 + quite cumbersome for users (the same setting has to be provided in multiple + places, making it easy to misconfigure one thing or another). + + Alternatively, the subchart's templates could be customized to read global.* + values instead. However, this means upgrading the subchart is quite burdensome + since the customizations have to be re-applied to the upgraded chart. This is + even less tenable with the frequency with which chart updates are needed. + + With this in mind, this template was specially crafted to be able to read K10 + values and update the values that will be passed to the subchart. + + --- + + To accomplish this, Helm's template parsing and rendering order is exploited. + + Helm allows parent charts to override templates in subcharts. This is done by + parsing templates with lower precedence first (templates that are more deeply + nested than others). This allows templates with higher precedence to redefine + templates with lower precedence. + + Helm also renders templates in this same order. This template exploits this + ordering in order to set subchart values before the subchart's templates are + rendered, having the same effect as the user setting the values. + + WARNING: The name and directory structure of this template was carefully + selected to ensure that it is rendered before other templates! +*/}} + +{{- if .Values.prometheus.server.enabled }} +{{- $prometheus_scoped_values := (dict "Chart" (dict "Name" "prometheus") "Release" .Release "Values" .Values.prometheus) -}} + +{{- $prometheus_name := (include "prometheus.name" $prometheus_scoped_values) -}} +{{- $prometheus_prefix := "/k10/prometheus/" -}} +{{- $release_name := .Release.Name -}} + +{{- /*** PROMETHEUS LABELS ***/ -}} +{{- $_ := mergeOverwrite .Values.prometheus + (dict + "commonMetaLabels" (dict + "app.kubernetes.io/name" $prometheus_name + "app.kubernetes.io/instance" $release_name + ) + ) +-}} + +{{- /*** PROMETHEUS SERVER OVERRIDES ***/ -}} +{{- $fullnameOverride := .Values.prometheus.server.fullnameOverride | default "prometheus-server" -}} +{{- $clusterRoleNameOverride := .Values.prometheus.server.clusterRoleNameOverride | default (printf "%s-%s" .Release.Name $fullnameOverride) -}} + +{{- /* Merge global pod labels with any prometheus-specific labels, where the latter is of highest priority */ -}} +{{- $podLabels := merge (dict) (.Values.prometheus.server.podLabels | default dict) (.Values.global.podLabels) -}} + +{{- /* Merge global pod labels with any prometheus-specific annotations, where the latter is of highest priority */ -}} +{{- $podAnnotations := merge (dict) (.Values.prometheus.server.podAnnotations | default dict) (.Values.global.podAnnotations) -}} +{{- if .Values.scc.create -}} + {{- $podAnnotations = merge (dict "openshift.io/required-scc" (printf "%s-prometheus-server" .Release.Name)) $podAnnotations -}} +{{- end -}} +{{- $_ := mergeOverwrite .Values.prometheus.server + (dict + "baseURL" (.Values.prometheus.server.baseURL | default $prometheus_prefix) + "prefixURL" (.Values.prometheus.server.prefixURL | default $prometheus_prefix | trimSuffix "/") + + "clusterRoleNameOverride" $clusterRoleNameOverride + "configMapOverrideName" "k10-prometheus-config" + "fullnameOverride" $fullnameOverride + "podLabels" $podLabels + "podAnnotations" $podAnnotations + ) +-}} + +{{- /*** K10 PROMETHEUS CONFIGMAP-RELOAD IMAGE *** + - global.airgapped.repository + - global.image.registry + - global.image.tag + - global.images.configmap-reload +*/ -}} +{{- $prometheus_configmap_reload_image := (dict + "registry" (.Values.global.airgapped.repository | default .Values.global.image.registry) + "repository" "configmap-reload" + "tag" (include "get.k10ImageTag" $) +) -}} + +{{- if (index .Values.global.images "configmap-reload") -}} + {{- $prometheus_configmap_reload_image = ( + include "k10.splitImage" (dict + "image" (index .Values.global.images "configmap-reload") + "path" "global.images.configmap-reload" + ) + ) | fromJson + -}} +{{- end -}} + +{{- if .Values.global.azMarketPlace -}} + {{- $prometheus_configmap_reload_image = (dict + "registry" .Values.global.azure.images.configmapreload.registry + "repository" .Values.global.azure.images.configmapreload.image + "tag" .Values.global.azure.images.configmapreload.tag + ) + -}} +{{- end -}} + +{{- $_ := mergeOverwrite .Values.prometheus.configmapReload.prometheus.image + (dict + "repository" (list $prometheus_configmap_reload_image.registry $prometheus_configmap_reload_image.repository | compact | join "/") + "tag" $prometheus_configmap_reload_image.tag + "digest" $prometheus_configmap_reload_image.digest + ) +-}} + +{{- /*** K10 PROMETHEUS SERVER IMAGE *** + - global.airgapped.repository + - global.image.registry + - global.image.tag + - global.images.prometheus +*/ -}} +{{- $prometheus_server_image := (dict + "registry" (.Values.global.airgapped.repository | default .Values.global.image.registry) + "repository" "prometheus" + "tag" (include "get.k10ImageTag" $) +) -}} +{{- if .Values.global.images.prometheus -}} + {{- $prometheus_server_image = ( + include "k10.splitImage" (dict + "image" .Values.global.images.prometheus + "path" "global.images.prometheus" + ) + ) | fromJson + -}} +{{- end -}} + +{{- if .Values.global.azMarketPlace -}} + {{- $prometheus_server_image = ( dict + "registry" .Values.global.azure.images.prometheus.registry + "repository" .Values.global.azure.images.prometheus.image + "tag" .Values.global.azure.images.prometheus.tag + ) + -}} +{{- end -}} + +{{- $_ := mergeOverwrite .Values.prometheus.server.image + (dict + "repository" (list $prometheus_server_image.registry $prometheus_server_image.repository | compact | join "/") + "tag" $prometheus_server_image.tag + "digest" $prometheus_server_image.digest + ) +-}} + +{{- /*** K10 IMAGE PULL SECRETS *** + - secrets.dockerConfig + - secrets.dockerConfigPath + - global.imagePullSecret +*/ -}} +{{- $image_pull_secret_names := list -}} +{{- if .Values.global.imagePullSecret -}} + {{- $image_pull_secret_names = append $image_pull_secret_names .Values.global.imagePullSecret -}} +{{- end -}} +{{- if (or .Values.secrets.dockerConfig .Values.secrets.dockerConfigPath) -}} + {{ $image_pull_secret_names = append $image_pull_secret_names "k10-ecr" -}} +{{- end -}} +{{- $image_pull_secret_names = $image_pull_secret_names | compact | uniq -}} + +{{- if $image_pull_secret_names -}} + {{- $image_pull_secrets := .Values.prometheus.imagePullSecrets | default list -}} + {{- range $name := $image_pull_secret_names -}} + {{- $image_pull_secrets = append $image_pull_secrets (dict "name" $name) -}} + {{- end -}} + {{- $_ := set .Values.prometheus "imagePullSecrets" $image_pull_secrets -}} +{{- end -}} + +{{- /*** K10 PERSISTENCE *** + - global.persistence.storageClass +*/ -}} +{{- $_ := mergeOverwrite .Values.prometheus.server.persistentVolume + (dict + "storageClass" (.Values.prometheus.server.persistentVolume.storageClass | default .Values.global.persistence.storageClass) + ) +-}} +{{- end }} diff --git a/charts/kasten/k10/7.0.1401/triallicense b/charts/kasten/k10/7.0.1401/triallicense new file mode 100644 index 000000000..cfe6dd46b --- /dev/null +++ b/charts/kasten/k10/7.0.1401/triallicense @@ -0,0 +1 @@ +Y3VzdG9tZXJOYW1lOiB0cmlhbHN0YXJ0ZXItbGljZW5zZQpkYXRlRW5kOiAnMjEwMC0wMS0wMVQwMDowMDowMC4wMDBaJwpkYXRlU3RhcnQ6ICcyMDIwLTAxLTAxVDAwOjAwOjAwLjAwMFonCmZlYXR1cmVzOgogIHRyaWFsOiBudWxsCmlkOiB0cmlhbC0wOWY4MzE5Zi0xODBmLTRhOTAtOTE3My1kOTJiNzZmMTgzNWUKcHJvZHVjdDogSzEwCnJlc3RyaWN0aW9uczoKICBub2RlczogNTAwCnNlcnZpY2VBY2NvdW50S2V5OiBudWxsCnZlcnNpb246IHYxLjAuMApzaWduYXR1cmU6IEYxbnVLUFV5STJtbDJGMmV1VHdGOXNZRTZMVU5rQ3ZiR2tTV1ZkT0ZqdERCb1B6SjUyVWFsVkFmRjVmQUxpcm5BcVhkcERnYi9YcnpxSEYrTE0xS2pEMVdXUFd0ZUdXNFc1anBPSFN0T296Y0c5M0pUUHF5M2l6TVk3RmczZVFLYTZzWDhBdnFwOXArWXVBMWNwTENlQ2dsR2dnOTVzSUFmYmRMMTBmV2d2RmR6QUt4dUZLN2psRzVtbG1CRVF5R0hrYWdoZFIrVGxzeUNTNEFkbXVBOEZodVUwZnRBdXN0b1M3R2JKd1BuTFI3STFZY1Q4OW8wU2xRZEJ2Yjg2QzdKbm1OdnY0aHhiSUo5TTJvWGJPSnQ4ZnBNcjhNWFR6YWRMTWJzSndhZ3VBVHlNUWF2cExHNXRPb0U2ZE1uMVlFVDZLdWZiYy9NdThVRDVYYXlDYTdkZz09Cg== diff --git a/charts/kasten/k10/7.0.1401/values.schema.json b/charts/kasten/k10/7.0.1401/values.schema.json new file mode 100644 index 000000000..04188d11f --- /dev/null +++ b/charts/kasten/k10/7.0.1401/values.schema.json @@ -0,0 +1,3126 @@ +{ + "$schema": "https://json-schema.org/draft/2019-09/schema", + "type": "object", + "properties": { + "rbac": { + "type": "object", + "title": "RBAC configuration", + "description": "Create RBAC seetings", + "properties": { + "create": { + "title": "Enable RBAC creation", + "description": "Toggle RBAC resource creation", + "type": "boolean", + "default": true + } + } + }, + "serviceAccount": { + "type": "object", + "title": "ServiceAccount details", + "description": "Configure ServiceAccount", + "properties": { + "create": { + "type": "boolean", + "default": true, + "title": "Create a ServiceAccount", + "description": "Specifies whether a ServiceAccount should be created" + }, + "name": { + "type": "string", + "default": "", + "title": "The name of the ServiceAccount", + "description": "The name of the ServiceAccount to use. If not set and create is true, a name is derived using the release and chart names" + } + } + }, + "scc": { + "type": "object", + "title": "Security Context Constraints details", + "description": "Configure Security Context Constraints", + "properties": { + "create": { + "type": "boolean", + "default": false, + "title": "Create K10 SSC", + "description": "Whether to create a SecurityContextConstraints for K10 ServiceAccounts" + }, + "priority": { + "type": "integer", + "default": 0, + "title": "SCC priority", + "description": "Sets the SecurityContextConstraints priority" + } + } + }, + "networkPolicy": { + "type": "object", + "title": "NetworkPolicy details", + "description": "Configure NetworkPolicy", + "properties": { + "create": { + "type": "boolean", + "default": true, + "title": "Create NetworkPolicies", + "description": "Whether to create NetworkPolicies for the K10 services" + } + } + }, + "global": { + "type": "object", + "title": "Global settings", + "properties": { + "image": { + "type": "object", + "title": "K10 image configurations", + "description": "Change K10 image settings", + "properties": { + "registry": { + "type": "string", + "default": "gcr.io/kasten-images", + "title": "K10 image registry", + "description": "Change default K10 image registry" + }, + "tag": { + "type": "string", + "default": "", + "title": "K10 image tag", + "description": "Change default K10 tag" + }, + "pullPolicy": { + "type": "string", + "default": "Always", + "title": "Container images pullPolicy", + "description": "Change default pullPolicy for all the images", + "enum": [ + "IfNotPresent", + "Always", + "Never" + ] + } + } + }, + "airgapped": { + "type": "object", + "title": "Airgapped offline installation", + "description": "Configure Airgapped offline installation", + "properties": { + "repository": { + "type": "string", + "default": "", + "title": "helm repository", + "description": "The helm repository for offline (airgapped) installation" + } + } + }, + "persistence": { + "type": "object", + "title": "Persistent Volume global details", + "description": "Configure global settings for Persistent Volume", + "properties": { + "mountPath": { + "type": "string", + "default": "/mnt/k10state", + "title": "Persistent Volume global mount path", + "description": "Change default path for Persistent Volume mount" + }, + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable Persistent Volume", + "description": "Create Persistent Volumes" + }, + "storageClass": { + "type": "string", + "default": "", + "title": "Persistent Volume global Storageclass", + "description": "If set to '-', dynamic provisioning is disabled. If undefined (the default) or set to null, the default provisioner is used. (e.g gp2 on AWS, standard on GKE, AWS & OpenStack)" + }, + "accessMode": { + "type": "string", + "default": "ReadWriteOnce", + "title": "Persistent Volume global AccessMode", + "description": "Change default AccessMode for Persistent Volumes", + "enum": [ + "ReadWriteOnce", + "ReadOnlyMany", + "ReadWriteMany" + ] + }, + "size": { + "type": "string", + "default": "20Gi", + "title": "Persistent Volume size", + "description": "Change default size for Persistent Volumes" + }, + "metering": { + "type": "object", + "title": "Metering service Persistent Volume details", + "description": "Configure Persistence Volume for metering service", + "properties": { + "size": { + "type": "string", + "default": "2Gi", + "title": "Metering service Persistent Volume size", + "description": "If not set, global.persistence.size is used" + } + } + }, + "catalog": { + "type": "object", + "title": "Catalog service Persistent Volume details", + "description": "Configure Persistence Volume for catalog service", + "properties": { + "size": { + "type": "string", + "default": "", + "title": "Catalog service Persistent Volume size", + "description": "If not set, global.persistence.size is used." + } + } + }, + "jobs": { + "type": "object", + "title": "Jobs service Persistent Volume details", + "description": "Configure Persistence Volume for jobs service", + "properties": { + "size": { + "type": "string", + "default": "", + "title": "Jobs service Persistent Volume size", + "description": "If not set, global.persistence.size is used." + } + } + }, + "logging": { + "type": "object", + "title": "Logging service Persistent Volume details", + "description": "Configure Persistence Volume for logging service", + "properties": { + "size": { + "type": "string", + "default": "", + "title": "Logging service Persistent Volume size", + "description": "If not set, global.persistence.size is used." + } + } + }, + "grafana": { + "type": "object", + "title": "Grafana service Persistent Volume details", + "description": "Configure Persistence Volume for grafana service", + "properties": { + "size": { + "type": "string", + "default": "5Gi", + "title": "Grafana service Persistent Volume size", + "description": "If not set, global.persistence.size is used." + } + } + } + } + }, + "podLabels": { + "type": "object", + "default": {}, + "title": "Custom labels to be set to all Kasten pods", + "description": "Configures custom pod labels to be set to all Kasten pods.", + "examples": [ + { + "foo": "bar" + } + ] + }, + "podAnnotations": { + "type": "object", + "default": {}, + "title": "Custom annotations to be set to all Kasten pods", + "description": "Configures custom pod annotations to be set to all Kasten pods.", + "examples": [ + { + "foo": "bar" + } + ] + }, + "rhMarketPlace": { + "type": "boolean", + "default": false, + "title": "RedHat marketplace config", + "description": "Set it to true while generating helm operator" + }, + "images": { + "type": "object", + "title": "Global image settings", + "properties": { + "aggregatedapis": { + "type": "string", + "default": "", + "title": "Aggregatedapis service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "auth": { + "type": "string", + "default": "", + "title": "Auth service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "bloblifecyclemanager": { + "type": "string", + "default": "", + "title": "Bloblifecyclemanager service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "catalog": { + "type": "string", + "default": "", + "title": "Catalog service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "configmap-reload": { + "type": "string", + "title": "Configmap-reload service container image", + "default": "", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "controllermanager": { + "type": "string", + "default": "", + "title": "Controllermanager service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "crypto": { + "type": "string", + "default": "", + "title": "Crypto service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "dashboardbff": { + "type": "string", + "default": "", + "title": "Dashboardbff service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "datamover": { + "type": "string", + "default": "", + "title": "Datamover service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "dex": { + "type": "string", + "default": "", + "title": "Dex service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "emissary": { + "type": "string", + "default": "", + "title": "Emissary service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "events": { + "type": "string", + "default": "", + "title": "Events service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "executor": { + "type": "string", + "default": "", + "title": "Executor service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "frontend": { + "type": "string", + "default": "", + "title": "Frontend service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "gateway": { + "type": "string", + "default": "", + "title": "Gateway service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "grafana": { + "type": "string", + "title": "Grafana service container image", + "default": "", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "init": { + "type": "string", + "title": "Generic init container image", + "default": "", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "jobs": { + "type": "string", + "default": "", + "title": "Jobs service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "kanister-tools": { + "type": "string", + "default": "", + "title": "Kanister-tools service container image", + "description": "Kanister-tools service container image contains set of tools, required for all kanister related operations. It is used for debug, troubleshooting, primer purposes as well" + }, + "kanister": { + "type": "string", + "default": "", + "title": "Kanister service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "k10tools": { + "type": "string", + "default": "", + "title": "k10tools service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "logging": { + "type": "string", + "default": "", + "title": "Logging service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "metering": { + "type": "string", + "default": "", + "title": "Metering service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "ocpconsoleplugin": { + "type": "string", + "default": "", + "title": "OpenShift Console Plugin container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "paygo_daemonset": { + "type": "string", + "default": "", + "title": "Paygo_daemonset service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "prometheus": { + "type": "string", + "default": "", + "title": "Prometheus service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "repositories": { + "type": "string", + "default": "", + "title": "Repositories service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "state": { + "type": "string", + "default": "", + "title": "State service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "upgrade": { + "type": "string", + "default": "", + "title": "Upgrade service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes. If not set, the image name is formed with '(global.airgapped.repository)|(global.image.registry)/:(Chart.AppVersion)|(image.tag)'" + }, + "vbrintegrationapi": { + "type": "string", + "default": "", + "title": "Vbrintegrationapi service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "garbagecollector": { + "type": "string", + "default": "", + "title": "Garbagecollector service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + }, + "metric-sidecar": { + "type": "string", + "default": "", + "title": "Metric-sidecar service container image", + "description": "Used for packaging RedHat Operator. Setting this flag along with global.rhMarketPlace=true overrides the default image name. This flag is only for internal purposes." + } + } + }, + "imagePullSecret": { + "type": "string", + "default": "", + "title": "Container image pull secret", + "description": "Secret which contains docker config for private repository. Use `k10-ecr` when secrets.dockerConfigPath is used." + }, + "prometheus": { + "type": "object", + "title": "Prometheus settings", + "description": "Global prometheus settings", + "properties": { + "external": { + "type": "object", + "title": "External prometheus settings", + "description": "Configure prometheus", + "properties": { + "host": { + "type": "string", + "default": "", + "title": "External prometheus host name", + "description": "Set prometheus host name" + }, + "port": { + "type": "string", + "default": "", + "title": "External prometheus port number", + "description": "Set prometheus port number" + }, + "baseURL": { + "type": "string", + "default": "", + "title": "External prometheus baseURL", + "description": "Set prometheus baseURL" + } + } + } + } + }, + "network": { + "type": "object", + "title": "Network settings", + "description": "Global network settings", + "properties": { + "enable_ipv6": { + "type": "boolean", + "default": false, + "title": "Enable ipv6", + "description": "Set true to enable ipv6" + } + } + } + } + }, + "route": { + "type": "object", + "title": "OpenShift route configuration", + "description": "Configure OpenShift Route", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Exposed dashboard via route", + "description": "Whether the K10 dashboard should be exposed via route" + }, + "host": { + "type": "string", + "default": "", + "title": "Host name", + "description": "Set Host name for the route" + }, + "path": { + "type": "string", + "default": "", + "title": "Route path", + "description": "Set Path for the route" + }, + "annotations": { + "type": "object", + "default": {}, + "title": "Route annotations", + "description": "Set annotations for the route", + "examples": [ + { + "kubernetes.io/tls-acme": "true", + "haproxy.router.openshift.io/disable_cookies": "true", + "haproxy.router.openshift.io/balance": "roundrobin" + } + ] + }, + "labels": { + "type": "object", + "default": {}, + "title": "Route label", + "description": "Set Labels for the route resource", + "examples": [ + { + "foo": "bar" + } + ] + }, + "tls": { + "type": "object", + "title": "Route TLS configuration", + "description": "Set TLS configuration for the route", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable TLS", + "description": "Whether to enable TLS" + }, + "insecureEdgeTerminationPolicy": { + "type": "string", + "default": "Redirect", + "title": "Route Termination Policy", + "description": "What to do in case of an insecure traffic edge termination", + "enum": [ + "None", + "Allow", + "Redirect", + "" + ] + }, + "termination": { + "type": "string", + "default": "edge", + "title": "Termination Schema", + "description": "Set termination Schema", + "enum": [ + "edge", + "passthrough", + "reencrypt" + ] + } + } + } + } + }, + "dexImage": { + "type": "object", + "title": "Dex image config", + "description": "Specify Dex image config", + "properties": { + "registry": { + "type": "string", + "default": "ghcr.io", + "title": "Dex image registry", + "description": "Change default image registry for Dex images" + }, + "repository": { + "type": "string", + "default": "dexidp", + "title": "Dex image repository", + "description": "Change default image repository for Dex images" + }, + "image": { + "type": "string", + "default": "dex", + "title": "Dex image name", + "description": "Change default image name for Dex images" + } + } + }, + "kanisterToolsImage": { + "type": "object", + "title": "kanister tools image config", + "description": "Set kanister tools image config", + "properties": { + "registry": { + "type": "string", + "default": "ghcr.io", + "title": "kanister-tools image registry", + "description": "Change default image registry for kanister-tools images" + }, + "repository": { + "type": "string", + "default": "kanisterio", + "title": "kanister-tools image repository", + "description": "Change default image repository for kanister-tools images" + }, + "image": { + "type": "string", + "default": "kanister-tools", + "title": "Kanister tools image name", + "description": "Change default image name for kanister-tools images" + }, + "pullPolicy": { + "type": "string", + "default": "Always", + "title": "Kanister tools image pullPolicy", + "description": "Change kanister-tools image pullPolicy", + "enum": [ + "IfNotPresent", + "Always", + "Never" + ] + } + } + }, + "ingress": { + "type": "object", + "title": "Ingress configuration", + "description": "Add ingress resource configuration", + "properties": { + "annotations": { + "type": "object", + "default": {}, + "title": "Ingress annotations", + "description": "Add optional annotations to the Ingress resource" + }, + "create": { + "type": "boolean", + "default": false, + "title": "Expose dashboard via ingress", + "description": "whether the K10 dashboard should be exposed via ingress" + }, + "tls": { + "type": "object", + "title": "TLS configuration for ingress", + "description": "Set TLS configuration for ingress", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable TLS", + "description": "Configures a TLS use for ingress.host" + }, + "secretName": { + "type": "string", + "default": "", + "title": "Optional TLS secret name", + "description": "Specifies the name of the secret to configure ingress.tls[].secretName" + } + } + }, + "name": { + "type": "string", + "default": "", + "title": "Ingress name", + "description": "Optional name of the Ingress object for the K10 dashboard." + }, + "class": { + "type": "string", + "default": "", + "title": "Ingress controller class", + "description": "Cluster ingress controller class: nginx, GCE" + }, + "host": { + "type": "string", + "default": "", + "title": "Ingress host name", + "description": "FQDN for name-based virtual host", + "examples": [ + "/k10.example.com" + ] + }, + "urlPath": { + "type": "string", + "default": "", + "title": "Ingress URL path", + "description": "URL path for K10 Dashboard", + "examples": [ + "/k10" + ] + }, + "pathType": { + "type": "string", + "default": "ImplementationSpecific", + "title": "Ingress path type", + "description": "Set the path type for the ingress resource", + "enum": [ + "Exact", + "Prefix", + "ImplementationSpecific" + ] + }, + "defaultBackend": { + "type": "object", + "title": "Ingress default backend", + "description": "Optional default backend for the Ingress object.", + "properties": { + "service": { + "type": "object", + "title": "Ingress default backend service", + "description": "A service referenced by the default backend (mutually exclusive with `resource`).", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable service default backend.", + "description": "Enable the default backend backed by a service." + }, + "name": { + "type": "string", + "default": "", + "title": "Service name", + "description": "Name of a service referenced by the default backend." + }, + "port": { + "type": "object", + "title": "Service port", + "description": "A port of a service referenced by the default backend.", + "properties": { + "name": { + "type": "string", + "default": "", + "title": "Port name", + "description": "Port name of a service referenced by the default backend (mutually exclusive with `number`)." + }, + "number": { + "type": "integer", + "default": 0, + "title": "Port number", + "description": "Port number of a service referenced by the default backend (mutually exclusive with `name`)." + } + } + } + } + }, + "resource": { + "type": "object", + "title": "Ingress default backend resource", + "description": "A resource referenced by the default backend (mutually exclusive with `service`).", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable resource default backend.", + "description": "Enable the default backend backed by a resource." + }, + "apiGroup": { + "type": "string", + "default": "", + "title": "Resource API group", + "description": "Optional API group of a resource referenced by the default backend.", + "examples": [ + "k8s.example.com" + ] + }, + "kind": { + "type": "string", + "default": "", + "title": "Resource kind", + "description": "Type of a resource referenced by the default backend.", + "examples": [ + "StorageBucket" + ] + }, + "name": { + "type": "string", + "default": "", + "title": "Resource name", + "description": "Name of a resource referenced by the default backend." + } + } + } + } + } + } + }, + "eula": { + "type": "object", + "title": "EULA configuration", + "properties": { + "accept": { + "type": "boolean", + "default": false, + "title": "Enable accept EULA before installation", + "description": "An End-User license agreement (EULA) is a legal agreement that grants a user a license to use an application or software. Users must consent to the EULA before purchasing, installing, or downloading an application or software owned by the service provider." + } + } + }, + "license": { + "type": "string", + "default": "", + "title": "License from Kasten", + "description": "Add license string obtained from Kasten" + }, + "cluster": { + "type": "object", + "title": "Cluster configuration", + "description": "Set cluster configuration", + "properties": { + "domainName": { + "type": "string", + "default": "", + "title": "Domain name of the cluster", + "description": "Set domain name of the cluster" + } + } + }, + "multicluster": { + "type": "object", + "title": "Multi-cluster configuration", + "description": "Configure the multi-cluster system", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable the multi-cluster system", + "description": "Choose whether to enable the multi-cluster system components and capabilities" + }, + "primary": { + "type": "object", + "title": "Multi-cluster primary configuration", + "description": "Configure multi-cluster primary", + "properties": { + "create": { + "type": "boolean", + "default": false, + "title": "Setup cluster as a multi-cluster primary", + "description": "Choose whether to setup cluster as a multi-cluster primary" + }, + "name": { + "type": "string", + "default": "", + "title": "Primary cluster name", + "description": "Choose the cluster name for multi-cluster primary" + }, + "ingressURL": { + "type": "string", + "default": "", + "title": "Primary cluster dashboard URL", + "description": "Choose the dashboard URL for the multi-cluster primary; e.g. https://cluster-name.domain/k10" + } + } + } + } + }, + "prometheus": { + "type": "object", + "title": "Internal Prometheus configuration", + "description": "Configure internal Prometheus", + "properties": { + "rbac": { + "type": "object", + "title": "Prometheus rbac", + "description": "Configure Prometheus rbac resources", + "properties": { + "create": { + "type": "boolean", + "default": false, + "title": "Enable Prometheus rbac. Warning - cluster wide permissions", + "description": "Choose whether to create Prometheus RBAC configuration. Warning: Enabling this action will allow Prometheus permission to scrape pods in all K8s namespaces." + } + } + }, + "server": { + "type": "object", + "title": "Prometheus Server", + "description": "Configure Prometheus Server", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable Prometheus server", + "description": "Create Prometheus server" + }, + "securityContext": { + "type": "object", + "title": "Prometheus server securityContext", + "description": "Configure Prometheus server securityContext", + "properties": { + "runAsUser": { + "type": "integer", + "default": 65534, + "title": "runAsUser ID", + "description": "Set securityContext runAsUser ID" + }, + "runAsNonRoot": { + "type": "boolean", + "default": true, + "title": "Enable runAsNonRoot", + "description": "Enable securityContext runAsNonRoot" + }, + "runAsGroup": { + "type": "integer", + "default": 65534, + "title": "runAsGroup ID", + "description": "Set securityContext runAsGroup ID" + }, + "fsGroup": { + "type": "integer", + "default": 65534, + "title": "fsGroup ID", + "description": "Set securityContext fsGroup ID" + } + } + }, + "retention": { + "type": "string", + "default": "30d", + "title": "Prometheus retention", + "description": "Set retention period for Prometheus" + }, + "persistentVolume": { + "type": "object", + "title": "Prometheus persistent volume", + "description": "Configure Prometheus persistent volume", + "properties": { + "storageClass": { + "type": "string", + "default": "", + "title": "StorageClassName used to create Prometheus PVC", + "description": "Setting this option overwrites global StorageClass value" + } + } + }, + "fullnameOverride": { + "type": "string", + "default": "prometheus-server", + "title": "Prometheus server deployment name", + "description": "Override default Prometheus server deployment name" + }, + "baseURL": { + "type": "string", + "default": "/k10/prometheus/", + "title": "Prometheus external url path", + "description": "Prometheus external url path at which the server can be accessed" + }, + "prefixURL": { + "type": "string", + "default": "/k10/prometheus", + "title": "Prometheus prefix slug", + "description": "Prometheus prefix slug at which the server can be accessed" + } + } + } + } + }, + "jaeger": { + "type": "object", + "title": "Jaeger configuration", + "description": "Jaeger tracing settings", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable Jaeger tracing", + "description": "Set true to enable Jaeger tracing" + }, + "agentDNS": { + "type": "string", + "default": "", + "title": "Jaeger agentDNS", + "description": "Set agentDNS for Jaeger tracing" + } + } + }, + "service": { + "type": "object", + "title": "K10 K8s services config", + "properties": { + "externalPort": { + "type": "integer", + "default": 8000, + "title": "externalPort for K10 services", + "description": "Override default 8000 externalPort for K10 services" + }, + "internalPort": { + "type": "integer", + "default": 8000, + "title": "internalPort for K10 services", + "description": "Override default 8000 internalPort for K10 services" + }, + "aggregatedApiPort": { + "type": "integer", + "default": 10250, + "title": "aggregatedApiPort for aggapi service", + "description": "Override default 10250 port for aggapi service" + }, + "gatewayAdminPort": { + "type": "integer", + "default": 8877, + "title": "Gateway admin port", + "description": "Override default 8877 gateway admin port" + } + } + }, + "secrets": { + "type": "object", + "title": "K10 secrets", + "description": "K10 secrets configuration", + "properties": { + "awsAccessKeyId": { + "type": "string", + "default": "", + "title": "AWS access key ID", + "description": "Set AWS access key ID required for AWS deployment" + }, + "awsSecretAccessKey": { + "type": "string", + "default": "", + "title": "AWS secret access key", + "description": "Set AWS access key secret" + }, + "awsIamRole": { + "type": "string", + "default": "", + "title": "AWS IAM Role", + "description": "ARN of the AWS IAM role assumed by K10 to perform any AWS operation" + }, + "awsClientSecretName": { + "type": "string", + "default": "", + "title": "Secret with AWS credentials and/or IAM Role", + "description": "Specify a Secret directly instead of having to provide awsAccessKeyId, awsSecretAccessKey and awsIamRole" + }, + "googleApiKey": { + "type": "string", + "default": "", + "title": "Google API Key", + "description": "Non-default base64 encoded GCP Service Account key" + }, + "googleProjectId": { + "type": "string", + "default": "", + "title": "Google Project ID", + "description": "Set Google Project ID other than the one in the GCP Service Account" + }, + "googleClientSecretName": { + "type": "string", + "default": "", + "title": "Secret with Google credentials", + "description": "Specify a Secret directly instead of having to provide googleApiKey and googleProjectId" + }, + "tlsSecret": { + "type": "string", + "default": "", + "title": "K8s TLS secret name contains for k10 Gateway service", + "description": "Specify a Secret directly instead of having to provide both the cert and key. This reduces the security risk a bit by not caching the certs and keys in the bash history." + }, + "dockerConfig": { + "type": "string", + "default": "", + "title": "Docker config", + "description": "base64 representation of your Docker credentials to pull docker images from a private registry" + }, + "dockerConfigPath": { + "type": "string", + "default": "", + "title": "Docker config path", + "description": "Path to Docker config file to create secret from" + }, + "azureTenantId": { + "type": "string", + "default": "", + "title": "Azure tenant ID", + "description": "Azure tenant ID required for Azure deployment" + }, + "azureClientId": { + "type": "string", + "default": "", + "title": "Azure client ID", + "description": "Azure Service App ID" + }, + "azureClientSecret": { + "type": "string", + "default": "", + "title": "Azure client Secret", + "description": "Azure Service APP secret" + }, + "azureClientSecretName": { + "type": "string", + "default": "", + "title": "Secret with Azure credentials", + "description": "Specify a Secret directly instead of having to provide azureClientId, azureTenantId and azureClientSecret" + }, + "azureResourceGroup": { + "type": "string", + "default": "", + "title": "Azure resource group", + "description": "Resource Group name that was created for the Kubernetes cluster" + }, + "azureSubscriptionID": { + "type": "string", + "default": "", + "title": "Azure subscription ID", + "description": "Subscription ID in your Azure tenant" + }, + "azureResourceMgrEndpoint": { + "type": "string", + "default": "", + "title": "Azure resource manager endpoint", + "description": "Resource management endpoint for the Azure Stack instance" + }, + "azureADEndpoint": { + "type": "string", + "default": "", + "title": "Azure AD endpoint", + "description": "Azure Active Directory login endpoint" + }, + "azureADResourceID": { + "type": "string", + "default": "", + "title": "Azure Active Directory resource ID", + "description": "Azure Active Directory resource ID to obtain AD tokens" + }, + "microsoftEntraIDEndpoint": { + "type": "string", + "default": "", + "title": "Microsoft Entra ID endpoint", + "description": "Microsoft Entra ID login endpoint" + }, + "microsoftEntraIDResourceID": { + "type": "string", + "default": "", + "title": "Microsoft Entra ID resource ID", + "description": "Microsoft Entra ID resource ID to obtain AD tokens" + }, + "azureCloudEnvID": { + "type": "string", + "default": "", + "title": "Azure Cloud Environment ID", + "description": "Azure Cloud Environment ID" + }, + "apiTlsCrt": { + "type": "string", + "default": "", + "title": "API TLS Certificate", + "description": "K8s API server TLS certificate" + }, + "apiTlsKey": { + "type": "string", + "default": "", + "title": "API TLS Key", + "description": "K8s API server TLS key" + }, + "vsphereEndpoint": { + "type": "string", + "default": "", + "title": "vSphere endpoint", + "description": "vSphere endpoint for login" + }, + "vsphereUsername": { + "type": "string", + "default": "", + "title": "", + "description": "" + }, + "vspherePassword": { + "type": "string", + "default": "", + "title": "vSphere password", + "description": "vSphere password for login" + }, + "vsphereClientSecretName": { + "type": "string", + "default": "", + "title": "Secret with vSphere credentials", + "description": "Specify a Secret directly instead of having to provide vsphereUsername, vspherePassword and vspherePassword" + } + } + }, + "metering": { + "type": "object", + "title": "Metering service config", + "description": "Metering service settings", + "properties": { + "reportingKey": { + "type": "string", + "default": "", + "title": "Reporting key", + "description": "Base64 encoded reporting key" + }, + "consumerId": { + "type": "string", + "default": "", + "title": "Consumer ID", + "description": "Consumer ID in the format project:" + }, + "awsRegion": { + "type": "string", + "default": "", + "title": "AWS Region", + "description": "Set AWS_REGION for metering service" + }, + "awsMarketPlaceIamRole": { + "type": "string", + "default": "", + "title": "AWS Marketplace IAM Role", + "description": "Set AWS marketplace IAM Role" + }, + "awsMarketplace": { + "type": "boolean", + "default": false, + "title": "AWS Marketplace", + "description": "Set AWS cloud metering license mode" + }, + "awsManagedLicense": { + "type": "boolean", + "default": false, + "title": "AWS managed license", + "description": "Set AWS managed license mode" + }, + "licenseConfigSecretName": { + "type": "string", + "default": "", + "title": "License config secret name", + "description": "AWS managed license config secret" + }, + "serviceAccount": { + "type": "object", + "title": "Metering service serviceAccount", + "description": "Configuration for metering service serviceAccount", + "properties": { + "create": { + "type": "boolean", + "default": false, + "title": "Create metering service serviceAccount", + "description": "Create metering service serviceAccount" + }, + "name": { + "type": "string", + "default": "", + "title": "Metering ServiceAccount name", + "description": "Set name for metering ServiceAccount" + } + } + }, + "mode": { + "type": "string", + "default": "", + "title": "Control license reporting", + "description": "Set to `airgap` for private-network installs" + }, + "redhatMarketplacePayg": { + "type": "boolean", + "default": false, + "title": "Red Hat cloud metering", + "description": "Set Red Hat cloud metering license mode" + }, + "reportCollectionPeriod": { + "type": "integer", + "default": 1800, + "title": "Report collection period", + "description": "Metric report collection period (in seconds)" + }, + "reportPushPeriod": { + "type": "integer", + "default": 3600, + "title": "Report push period", + "description": "Metric report push period (in seconds)" + }, + "promoID": { + "type": "string", + "default": "", + "title": "K10 promotion ID", + "description": "K10 promotion ID from marketing campaigns" + } + } + }, + "clusterName": { + "type": "string", + "default": "", + "title": "Cluster name", + "description": "Cluster name for better logs visibility" + }, + "executorReplicas": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Number of executor service pod replicas", + "description": "Deprecated. Please use 'limiter.executorReplicas' parameter" + }, + "logLevel": { + "type": "string", + "default": "info", + "title": "Log level", + "description": "Change default log level" + }, + "externalGateway": { + "type": "object", + "title": "External gateway", + "description": "Configure external gateway for K10 API services", + "properties": { + "create": { + "type": "boolean", + "default": false, + "title": "Enable external gateway", + "description": "Create external gateway service" + }, + "annotations": { + "type": "object", + "title": "The annotations Schema", + "default": {}, + "description": "Standard annotations for the services" + }, + "fqdn": { + "type": "object", + "title": "Host and domain name for the K10 API services", + "description": "Configure host and domain name for the K10 API services", + "properties": { + "name": { + "type": "string", + "default": "", + "title": "Domain name for the K10 API services", + "description": "Domain name for the K10 API services" + }, + "type": { + "type": "string", + "default": "", + "title": "Gateway type", + "description": "Supported gateway type: route53-mapper or external-dns", + "enum": ["", "route53-mapper", "external-dns"] + } + } + }, + "awsSSLCertARN": { + "type": "string", + "default": "", + "title": "AWS SSL Cert ARN", + "description": "ARN for the AWS ACM SSL certificate used in the K10 API server" + } + } + }, + "auth": { + "type": "object", + "title": "Authentication settings", + "description": "Configure K10 dashboard authentication", + "properties": { + "groupAllowList": { + "type": "array", + "default": [], + "items": { + "type": "string" + }, + "title": "List of groups allowed to access K10 dashboard", + "description": "A list of groups whose members are allowed access to K10's dashboard", + "examples": [ + [ + "group1", + "group2" + ] + ] + }, + "basicAuth": { + "type": "object", + "title": "Basic authentication for the K10 dashboard", + "description": "Configure basic authentication for the K10 dashboard", + "properties": { + "enabled": { + "title": "Enable basic authentication", + "description": "Enables basic authentication to the K10 dashboard that allows users to login with username and password", + "type": "boolean", + "default": false + }, + "secretName": { + "type": "string", + "default": "", + "title": "Secret with basic auth creds", + "description": "Name of an existing Secret that contains a file generated with htpasswd" + }, + "htpasswd": { + "type": "string", + "default": "", + "title": "Basic authentication creds", + "description": "A username and password pair separated by a colon character" + } + } + }, + "tokenAuth": { + "type": "object", + "title": "Token based authentication", + "description": "Configuration for Token based authentication for the K10 dashboard", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable token based authentication", + "description": "Enable token based authentication to access K10 dashboard" + } + } + }, + "oidcAuth": { + "type": "object", + "default": {}, + "title": "Open ID Connect based authentication", + "description": "Configuration for Open ID Connect based authentication for the K10 dashboard", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable Open ID Connect based authentication", + "description": "Enable Open ID Connect based authentication to access K10 dashboard" + }, + "providerURL": { + "type": "string", + "default": "", + "title": "OIDC Provider URL", + "description": "URL for the OIDC Provider" + }, + "redirectURL": { + "type": "string", + "default": "", + "title": "K10 gateway service URL", + "description": "URL to the K10 gateway service" + }, + "scopes": { + "type": "string", + "default": "", + "title": "OIDC scopes", + "description": "Space separated OIDC scopes required for userinfo", + "examples": [ + "profile email" + ] + }, + "prompt": { + "type": "string", + "title": "OIDC prompt type", + "description": "The type of prompt to be used during authentication", + "default": "select_account", + "enum": [ + "none", + "consent", + "login", + "select_account" + ] + }, + "clientID": { + "type": "string", + "default": "", + "title": "OIDC client ID", + "description": "Client ID given by the OIDC provider" + }, + "clientSecret": { + "type": "string", + "default": "", + "title": "OIDC client secret", + "description": "Client secret given by the OIDC provider" + }, + "clientSecretName": { + "type": "string", + "default": "", + "title": "Reference to secret", + "description": "Secret containing OIDC client ID and OIDC client secret" + }, + "usernameClaim": { + "type": "string", + "default": "", + "title": "OIDC username claim", + "description": "The claim to be used as the username" + }, + "usernamePrefix": { + "type": "string", + "default": "", + "title": "OIDC username prefix", + "description": "Prefix that has to be used with the username obtained from the username claim" + }, + "groupClaim": { + "type": "string", + "default": "", + "title": "OIDC group claim", + "description": "Name of a custom OpenID Connect claim for specifying user groups" + }, + "groupPrefix": { + "type": "string", + "default": "", + "title": "OIDC group prefix", + "description": "All groups will be prefixed with this value to prevent conflicts" + }, + "logoutURL": { + "type": "string", + "default": "", + "title": "OIDC logout endpoint", + "description": "URL to your OIDC provider's logout endpoint" + }, + "secretName": { + "type": "string", + "default": "", + "title": "OIDC config based existing secret", + "description": "Must include providerURL, redirectURL, scopes, clientID/secret and logoutURL" + }, + "sessionDuration": { + "type": "string", + "default": "1h", + "title": "OIDC session duration", + "description": "Maximum OIDC session duration. Default value is 1 hour" + }, + "refreshTokenSupport": { + "type": "boolean", + "default": false, + "title": "OIDC Refresh Token support", + "description": "Enable OIDC Refresh Token support. Disabled by default." + } + } + }, + "openshift": { + "type": "object", + "title": "OpenShift OAuth server based authentication", + "description": "OpenShift OAuth server based authentication for K10 dashboard", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable OpenShift OAuth server based authentication", + "description": "Enable OpenShift OAuth server based authentication to access K10 dashboard" + }, + "serviceAccount": { + "type": "string", + "default": "", + "title": "Service account that represents an OAuth client", + "description": "Name of the service account that represents an OAuth client" + }, + "clientSecret": { + "type": "string", + "default": "", + "title": "Service account token", + "description": "The token corresponding to the service account" + }, + "clientSecretName": { + "type": "string", + "default": "", + "title": "Service account token secret", + "description": "The secret that contains the token corresponding to the service account" + }, + "dashboardURL": { + "type": "string", + "default": "", + "title": "K10 dashboard URL", + "description": "The URL used for accessing K10's dashboard" + }, + "openshiftURL": { + "type": "string", + "default": "", + "title": "OpenShift URL", + "description": "The URL for accessing OpenShift's API server" + }, + "insecureCA": { + "type": "boolean", + "default": false, + "title": "Disable SSL verification of connections to OpenShift", + "description": "Set true to turn off SSL verification of connections to OpenShift" + }, + "useServiceAccountCA": { + "type": "boolean", + "default": false, + "title": "use the CA certificate corresponding to the Service Account", + "description": "Usually found at ``/var/run/secrets/kubernetes.io/serviceaccount/ca.crt``" + }, + "secretName": { + "type": "string", + "default": "", + "title": "The Kubernetes Secret that contains OIDC settings", + "description": "Specify Kubernetes Secret that contains OIDC settings" + }, + "usernameClaim": { + "type": "string", + "default": "email", + "title": "Username claim", + "description": "The claim to be used as the username" + }, + "usernamePrefix": { + "type": "string", + "default": "", + "title": "Username prefix", + "description": "Prefix that has to be used with the username obtained from the username claim" + }, + "groupnameClaim": { + "type": "string", + "default": "groups", + "title": "custom OpenID Connect claim name for specifying user groups", + "description": "Name of a custom OpenID Connect claim for specifying user groups" + }, + "groupnamePrefix": { + "type": "string", + "default": "", + "title": "User group name prefix", + "description": "Prefix for user group name" + }, + "caCertsAutoExtraction": { + "type": "boolean", + "default": true, + "title": "Enable the OCP CA certificates automatic extraction", + "description": "Enable the OCP CA certificates automatic extraction to the K10 namespace" + } + } + }, + "ldap": { + "type": "object", + "title": "Active Directory/LDAP based authentication ", + "description": "Active Directory/LDAP based authentication for the K10 dashboard", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable Active Directory/LDAP based authentication", + "description": "Enable Active Directory/LDAP based authentication to access K10 dashboard" + }, + "restartPod": { + "type": "boolean", + "default": false, + "title": "force a restart of the authentication service pod", + "description": "force a restart of the authentication service pod (useful when updating authentication config)" + }, + "dashboardURL": { + "type": "string", + "default": "", + "title": "K10 dashboard URL", + "description": "The URL used for accessing K10's dashboard" + }, + "host": { + "type": "string", + "default": "", + "title": "Host and port of the AD/LDAP server", + "description": "Host and optional port of the AD/LDAP server in the form `host:port`" + }, + "insecureNoSSL": { + "type": "boolean", + "default": false, + "title": "Insecure AD/LDAP host", + "description": "Set if the AD/LDAP host is not using TLS" + }, + "insecureSkipVerifySSL": { + "type": "boolean", + "default": false, + "title": "Skip SSL verification of connections to the AD/LDAP host", + "description": "Turn off SSL verification of connections to the AD/LDAP host" + }, + "startTLS": { + "type": "boolean", + "default": false, + "title": "TLS protocol", + "description": "When set to true, ldap:// is used to connect to the server followed by creation of a TLS session. When set to false, ldaps:// is used." + }, + "bindDN": { + "type": "string", + "default": "", + "title": "Username for connecting to the AD/LDAP host", + "description": "The Distinguished Name(username) used for connecting to the AD/LDAP host" + }, + "bindPW": { + "type": "string", + "default": "", + "title": "The password for `bindDN`", + "description": "The password corresponding to the `bindDN` for connecting to the AD/LDAP host" + }, + "bindPWSecretName": { + "type": "string", + "default": "", + "title": "Secret name containing the password", + "description": "Secret name containing the password corresponding to the `bindDN` for connecting to the AD/LDAP host" + }, + "userSearch": { + "type": "object", + "title": "User search config", + "description": "AD/LDAP user search config", + "properties": { + "baseDN": { + "type": "string", + "default": "", + "title": "The base username to start the AD/LDAP search from", + "description": "The base Distinguished Name to start the AD/LDAP search from" + }, + "filter": { + "type": "string", + "default": "", + "title": "filter to apply when searching", + "description": "Optional filter to apply when searching the directory" + }, + "username": { + "type": "string", + "default": "", + "title": "Username to search in the directory", + "description": "Attribute used for comparing user entries when searching the directory" + }, + "idAttr": { + "type": "string", + "default": "", + "title": "Attribute in a user's entry that should map to the user ID field in a token", + "description": "AD/LDAP attribute in a user's entry that should map to the user ID field in a token" + }, + "emailAttr": { + "type": "string", + "default": "", + "title": "Attribute in a user's entry that should map to the email field in a token", + "description": "AD/LDAP attribute in a user's entry that should map to the email field in a token" + }, + "nameAttr": { + "type": "string", + "default": "", + "title": "Attribute in a user's entry that should map to the name field in a token", + "description": "Attribute in a user's entry that should map to the name field in a token" + }, + "preferredUsernameAttr": { + "type": "string", + "default": "", + "title": "Attribute in a user's entry that should map to the preferred_username field in a token", + "description": "AD/LDAP attribute in a user's entry that should map to the preferred_username field in a token" + } + } + }, + "groupSearch": { + "type": "object", + "title": "AD/LDAP group search config", + "description": "AD/LDAP group search config", + "properties": { + "baseDN": { + "type": "string", + "default": "", + "title": "The base Distinguished Name", + "description": "The base Distinguished Name to start the AD/LDAP group search from" + }, + "filter": { + "type": "string", + "default": "", + "title": "Search filter", + "description": "filter to apply when searching the directory for groups" + }, + "userMatchers": { + "type": "array", + "items": { + "type": "object", + "properties": { + "userAttr": { + "type": "string", + "default": "", + "title": "Attribute in the user's entry", + "description": "Attribute in the user's entry that must match the groupAttr when searching for groups" + }, + "groupAttr": { + "type": "string", + "default": "", + "title": "Attribute in the group's entry", + "description": "Attribute in the group's entry that must match the userAttr when searching for groups" + } + } + }, + "default": [], + "title": "List of field pairs that are used to match a user to a group", + "description": "List of field pairs that are used to match a user to a group" + }, + "nameAttr": { + "type": "string", + "default": "", + "title": "Attribute that represents a group's name in the directory", + "description": "The AD/LDAP attribute that represents a group's name in the directory" + } + } + }, + "secretName": { + "type": "string", + "default": "", + "title": "The Kubernetes Secret with OIDC settings", + "description": "The Kubernetes Secret that contains OIDC settings" + }, + "usernameClaim": { + "type": "string", + "default": "email", + "title": "Username claim", + "description": "The claim to be used as the username" + }, + "usernamePrefix": { + "type": "string", + "default": "", + "title": "Username prefix", + "description": "Prefix that has to be used with the username obtained from the username claim" + }, + "groupnameClaim": { + "type": "string", + "default": "groups", + "title": "Name of a custom OpenID Connect claim for specifying user groups", + "description": "Name of a custom OpenID Connect claim for specifying user groups" + }, + "groupnamePrefix": { + "type": "string", + "default": "", + "title": "Group name prefix", + "description": "Prefix for user group name" + } + } + }, + "k10AdminUsers": { + "type": "array", + "items": { + "type": "string" + }, + "default": [], + "title": "Admin users list", + "description": "A list of users who are granted admin level access to K10's dashboard" + }, + "k10AdminGroups": { + "type": "array", + "items": { + "type": "string" + }, + "default": [], + "title": "Admin groups list", + "description": "A list of groups whose members are granted admin level access to K10's dashboard" + } + } + }, + "optionalColocatedServices": { + "type": "object", + "title": "Optional Colocated services config", + "description": "Settings to enable optional colocated services", + "properties": { + "vbrintegrationapi": { + "title": "VBRIntegratipnAPI service", + "description": "Settings for VBRIntegratipnAPI service", + "type": "object", + "properties": { + "enabled": { + "title": "Enable VBRIntegratipnAPI service", + "description": "Set true to enable VBRIntegratipnAPI service", + "type": "boolean", + "default": true + } + } + } + } + }, + "cacertconfigmap": { + "type": "object", + "title": "CA Certificate ConfigMap", + "description": "ConfigMap containing a certificate for a trusted root certificate authority", + "properties": { + "name": { + "title": "Name of the configmap", + "description": "Name of the K8s ConfigMap containing a certificate for a trusted root certificate authority", + "type": "string", + "default": "" + } + } + }, + "apiservices": { + "type": "object", + "title": "Skip APIService objects creation", + "describe": "Skip APIService objects creation if already exists", + "properties": { + "deployed": { + "type": "boolean", + "default": true, + "title": "Whether APIService object are deployed", + "description": "Set true if APIService objects exists. Setting false will recreate the objects" + } + } + }, + "injectKanisterSidecar": { + "type": "object", + "title": "Kanister sidecar injection for workload pods", + "description": "Configure Kanister sidecar injection for workload pods", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable Kanister sidecar injection for workload pods", + "description": "Set true to enable Kanister sidecar injection for workload pods" + }, + "namespaceSelector": { + "type": "object", + "title": "namespaceSelector config", + "description": "Configure namespaceSelector for namespace containing the workloads to inject Kansiter Sidecar", + "properties": { + "matchLabels": { + "type": "object", + "default": {}, + "title": "namespaceSelector matchLabels", + "description": "Set of labels to select namespaces in which sidecar injection is enabled for workloads" + } + } + }, + "objectSelector": { + "type": "object", + "title": "objectSelector config", + "description": "Configure objectSelector for the workloads to inject Kansiter Sidecar", + "properties": { + "matchLabels": { + "type": "object", + "default": {}, + "title": "objectSelector matchLabels", + "description": "Set of labels to filter workload objects in which the sidecar is injected" + } + } + }, + "webhookServer": { + "type": "object", + "title": "Sidecar injector webhook server", + "description": "Configure sidecar injector webhook server", + "properties": { + "port": { + "type": "integer", + "default": 8080, + "title": "Mutating webhook server port number", + "description": "Port number on which the mutating webhook server accepts request" + } + } + } + } + }, + "kanisterPodCustomLabels": { + "type": "string", + "default": "", + "title": "Kanister pod custom labels", + "description": "Custom labels for pods managed by Kanister" + }, + "kanisterPodCustomAnnotations": { + "type": "string", + "default": "", + "title": "Kanister pod custom annotations", + "description": "Custom annotations added to pods managed by Kanister" + }, + "features": { + "type": "object", + "title": "Feature flags", + "description": "Feature flags to be set by K10", + "properties": { + "backgroundMaintenanceRun": { + "type": "boolean", + "default": true, + "title": "Background maintenance feature", + "description": "Enable background maintenance runs by the repositories service" + } + } + }, + "kanisterPodMetricSidecar": { + "type": "object", + "deprecated": true, + "title": "Deprecated: Metric sidecar for ephemeral pods", + "description": "Deprecated. Use 'workerPodMetricSidecar' parameter instead", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable sidecar container", + "description": "Enable sidecar container for gathering metrics from ephemeral pods" + }, + "metricLifetime": { + "type": "string", + "default": "2m", + "title": "The period we check if there are metrics which should be removed", + "description": "The period we check if there are metrics which should be removed" + }, + "pushGatewayInterval": { + "type": "string", + "default": "30s", + "title": "Pushgateway metrics interval", + "description": "The interval of sending metrics into the Pushgateway" + }, + "resources": { + "type": "object", + "title": "Kanister pod metric sidecar resource config", + "description": "Configure resource requests and limits for kanister pod metric sidecar", + "properties": { + "requests": { + "type": "object", + "title": "Kanister pod metric sidecar resource requests", + "description": "Kanister pod metric sidecar resource requests configuration", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Kanister pod metric sidecar memory request", + "description": "Kanister pod metric sidecar memory request", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Kanister pod metric sidecars cpu request", + "description": "Kanister pod metric sidecars cpu request", + "examples": [ + "1" + ] + } + } + }, + "limits": { + "type": "object", + "title": "Kanister pod metric sidecar resource limits", + "description": "Kanister pod metric sidecar resource limits configuration", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Kanister pod metric sidecars memory limit", + "description": "Kanister pod metric sidecars memory limit", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Kanister pod metric sidecars cpu limit", + "description": "Kanister pod metric sidecars cpu limit", + "examples": [ + "1" + ] + } + } + } + } + } + } + }, + "workerPodMetricSidecar": { + "type": "object", + "title": "Metric sidecar for ephemeral pods", + "description": "Sidecar container for gathering metrics from ephemeral pods", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enables a sidecar container for temporary worker Pods used to push Pod performance metrics to Prometheus", + "description": "Enables a sidecar container for temporary worker Pods used to push Pod performance metrics to Prometheus" + }, + "metricLifetime": { + "type": "string", + "default": "2m", + "title": "Specifies the period after which metrics for an individual worker Pod are removed from Prometheus ", + "description": "Specifies the period after which metrics for an individual worker Pod are removed from Prometheus " + }, + "pushGatewayInterval": { + "type": "string", + "default": "30s", + "title": "Specifies the frequency for pushing metrics into Prometheus", + "description": "Specifies the frequency for pushing metrics into Prometheus" + }, + "resources": { + "type": "object", + "title": "Specifies resource requests and limits for the temporary worker Pod metric sidecar", + "description": "Specifies resource requests and limits for the temporary worker Pod metric sidecar", + "properties": { + "requests": { + "type": "object", + "title": "Specifies resource requests for the temporary worker Pod metric sidecar", + "description": "Specifies resource requests for the temporary worker Pod metric sidecar", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Temporary worker Pod metric sidecar memory request", + "description": "Temporary worker Pod metric sidecar memory request", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Temporary worker Pod metric sidecars cpu request", + "description": "Temporary worker Pod metric sidecars cpu request", + "examples": [ + "1" + ] + } + } + }, + "limits": { + "type": "object", + "title": "Specifies resource limits for the temporary worker Pod metric sidecar", + "description": "Specifies resource limits for the temporary worker Pod metric sidecar", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Temporary worker Pod metric sidecars memory limit", + "description": "Temporary worker Pod metric sidecars memory limit", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Temporary worker Pod metric sidecars cpu limit", + "description": "Temporary worker Pod metric sidecars cpu limit", + "examples": [ + "1" + ] + } + } + } + } + } + } + }, + "genericStorageBackup": { + "type": "object", + "title": "Generic Storage backup activation config", + "properties": { + "token": { + "type": "string", + "title": "Generic volume snapshot activation token", + "description": "Token to enable generic volume snapshot", + "default": "" + } + } + }, + "genericVolumeSnapshot": { + "type": "object", + "title": "Generic Volume Snapshot restore pods config", + "description": "Specifies resource requests and limits for generic backup sidecar and all temporary Kasten worker Pods. Superseded by ActionPodSpec", + "properties": { + "resources": { + "type": "object", + "title": "Generic Volume Snapshot restore pod resource config", + "description": "Specifies resource requests for generic backup sidecar and all temporary Kasten worker Pods. Superseded by ActionPodSpec", + "properties": { + "requests": { + "type": "object", + "title": "Generic Volume Snapshot resource requests", + "description": "Generic Volume Snapshot resource requests configuration", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Generic Volume Snapshot restore pods memory request", + "description": "Generic Volume Snapshot restore pods memory request", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Generic Volume Snapshot restore pods cpu request", + "description": "Generic Volume Snapshot restore pods cpu request", + "examples": [ + "1" + ] + } + } + }, + "limits": { + "type": "object", + "title": "Generic Volume Snapshot resource limits", + "description": "Specifies resource limits for generic backup sidecar and all temporary Kasten worker Pods. Superseded by ActionPodSpec", + "properties": { + "memory": { + "type": "string", + "default": "", + "title": "Generic Volume Snapshot restore pods memory limit", + "description": "Generic Volume Snapshot restore pods memory limit", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "", + "title": "Generic Volume Snapshot restore pods cpu limit", + "description": "Generic Volume Snapshot restore pods cpu limit", + "examples": [ + "1" + ] + } + } + } + } + } + } + }, + "garbagecollector": { + "type": "object", + "title": "garbage collection", + "description": "Configure garbage collection settings", + "properties": { + "daemonPeriod": { + "type": "integer", + "default": 21600, + "title": "Garbage collection period", + "description": "Set garbage collection period (in seconds)" + }, + "keepMaxActions": { + "type": "integer", + "default": 1000, + "title": "Max actions to keep", + "description": "Sets maximum actions to keep" + }, + "actions": { + "type": "object", + "title": "action collectors config", + "description": "Configure action garbage collectors", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable action collectors", + "description": "Set true to enable action collectors" + } + } + } + } + }, + "resources": { + "type": "object", + "default": {}, + "title": "K10 pods resource config", + "description": "Resource management for K10 pods" + }, + "datastore": { + "type": "object", + "properties": { + "parallelUploads": { + "type": "integer", + "default": 8, + "title": "Parallelism for data store uploads", + "description": "Specifies how many files can be uploaded in parallel to the data store" + }, + "parallelDownloads": { + "type": "integer", + "default": 8, + "title": "Parallelism for data store downloads", + "description": "Specifies how many files can be downloaded in parallel from the data store" + } + } + }, + "defaultPriorityClassName": { + "type": "string", + "default": "", + "title": "Default priorityClassName", + "description": "Set the default priorityClassName for all K10 pods" + }, + "priorityClassName": { + "type": "object", + "default": {}, + "title": "K10 pods priorityClassName config", + "description": "Set priorityClassName for specific K10 pods" + }, + "services": { + "type": "object", + "title": "K10 services config", + "description": "Settings for K10 services", + "properties": { + "executor": { + "type": "object", + "title": "executor service config", + "description": "Configuration for K10 executor service", + "properties": { + "hostNetwork": { + "type": "boolean", + "default": false, + "title": "Enable node network usage", + "description": "Whether the executor pods may use the node network" + }, + "workerCount": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Executor workers count", + "description": "Deprecated. Use 'limiter.executorThreads' parameter instead" + }, + "maxConcurrentRestoreCsiSnapshots": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent restore CSI snapshots operations", + "description": "Deprecated. Use 'limiter.csiSnapshotRestoresPerAction' parameter instead" + }, + "maxConcurrentRestoreGenericVolumeSnapshots": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent restore generic volume snapshots operations", + "description": "Deprecated. Use 'limiter.volumeRestoresPerAction' parameter instead" + }, + "maxConcurrentRestoreWorkloads": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent restore workloads operations", + "description": "Deprecated. Use 'limiter.workloadRestoresPerAction' parameter instead" + } + } + }, + "dashboardbff": { + "type": "object", + "title": "dashboardbff service config", + "properties": { + "hostNetwork": { + "type": "boolean", + "default": false, + "title": "Enable node network usage", + "description": "Whether the dashboardbff pods may use the node network" + } + } + }, + "securityContext": { + "type": "object", + "title": "securityContext for K10 service containers", + "description": "Custom securityContext for K10 service containers", + "properties": { + "runAsUser": { + "type": "integer", + "default": 1000, + "title": "runAsUser ID", + "description": "User ID K10 service containers run as" + }, + "fsGroup": { + "type": "integer", + "default": 1000, + "title": "FSGroup ID", + "description": "FSGroup that owns K10 service container volumes" + }, + "runAsNonRoot": { + "type": "boolean", + "default": true, + "title": "RunAsNonRoot", + "description": "Indicates that K10 service containers should run as non-root user." + }, + "seccompProfile": { + "type": "object", + "title": "Seccomp Profile object", + "description": "Sets the Seccomp profile for K10 service containers", + "properties": { + "type": { + "type": "string", + "default": "RuntimeDefault", + "title": "Seccomp profile type", + "description": "Sets the Seccomp profile type for K10 service containers" + } + } + } + } + }, + "aggregatedapis": { + "type": "object", + "title": "K10 aggregatedapis service config", + "properties": { + "hostNetwork": { + "type": "boolean", + "default": false, + "title": "Enable node network usage", + "description": "Whether the aggregatedapis pods may use the node network" + } + } + } + } + }, + "siem": { + "type": "object", + "title": "siem", + "description": "siem settings", + "properties": { + "logging": { + "type": "object", + "title": "logging", + "description": "siem logging settings", + "properties": { + "cluster": { + "type": "object", + "title": "cluster", + "description": "In-cluster agent log slurping settings", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable in-cluster agent-based audit logging", + "description": "Enabled in-cluster agent-based audit logging for K10 events" + } + } + }, + "cloud": { + "type": "object", + "title": "cloud", + "description": "siem cloud logging settings", + "properties": { + "path": { + "type": "string", + "default": "k10audit/", + "title": "Directory path in cloud object storage for saving logs", + "description": "Directory path in cloud object storage for saving logs when writing K10 events" + }, + "awsS3": { + "type": "object", + "title": "awsS3", + "description": "AWS S3 log slurping settings", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable AWS S3 audit logging", + "description": "Enable AWS S3 audit logging for K10 events" + } + } + } + } + } + } + } + } + }, + "apigateway": { + "type": "object", + "title": "APIGateway", + "description": "APIGateway settings", + "properties": { + "serviceResolver": { + "type": "string", + "default": "dns", + "title": "Resolver used for service discovery", + "description": "The resolver used for service discovery in the API gateway", + "enum": [ + "dns", + "endpoint" + ] + } + } + }, + "limiter": { + "type": "object", + "title": "Limiter", + "description": "Limits set on several operations", + "properties": { + "concurrentSnapConversions": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent snapshot conversions", + "description": "Deprecated. Use 'limiter.snapshotExportsPerAction' parameter instead" + }, + "snapshotExportsPerAction": { + "type": "integer", + "default": 3, + "title": "Per action limit of concurrent volume export operations", + "description": "Per action limit of concurrent volume export operations" + }, + "genericVolumeSnapshots": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent generic volume snapshot creation", + "description": "Deprecated. Use 'limiter.genericVolumeBackupsPerCluster' parameter instead." + }, + "genericVolumeBackupsPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent Generic Volume Backup operations", + "description": "Cluster-wide limit of concurrent Generic Volume Backup operations" + }, + "genericVolumeCopies": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent generic volume snapshot copy", + "description": "Deprecated. Use 'limiter.snapshotExportsPerCluster' parameter instead" + }, + "snapshotExportsPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent volume export operations", + "description": "Cluster-wide limit of concurrent volume export operations" + }, + "genericVolumeRestores": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent generic volume snapshot restore", + "description": "Deprecated. Use 'limiter.volumeRestoresPerCluster' parameter instead" + }, + "volumeRestoresPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent volume restore operations", + "description": "Cluster-wide limit of concurrent volume restore operations from exported backups" + }, + "csiSnapshots": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent CSI snapshot create", + "description": "Deprecated. Use 'limiter.csiSnapshotsPerCluster' parameter instead" + }, + "csiSnapshotsPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent CSI VolumeSnapshot creation requests", + "description": "Cluster-wide limit of concurrent CSI VolumeSnapshot creation requests" + }, + "providerSnapshots": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent cloud provider create", + "description": "Deprecated. Use 'limiter.directSnapshotsPerCluster' parameter instead" + }, + "directSnapshotsPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent non-CSI snapshot creation requests", + "description": "Cluster-wide limit of concurrent non-CSI snapshot creation requests" + }, + "imageCopies": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Concurrent image copy", + "description": "Deprecated. Use 'limiter.imageCopiesPerCluster' parameter instead" + }, + "imageCopiesPerCluster": { + "type": "integer", + "default": 10, + "title": "Cluster-wide limit of concurrent ImageStream container image operations", + "description": "Cluster-wide limit of concurrent ImageStream container image backup (i.e. copy from) and restore (i.e. copy to) operations" + }, + "executorReplicas": { + "type": "integer", + "default": 3, + "title": "Number of executor service pod replicas", + "description": "Specifies the number of executor-svc Pods used to process Kasten jobs" + }, + "executorThreads": { + "type": "integer", + "default": 8, + "title": "Executor threads count", + "description": "Specifies the number of threads per executor-svc Pod used to process Kasten jobs" + }, + "workloadRestoresPerAction": { + "type": "integer", + "default": 3, + "title": "Per action limit of concurrent manifest data restores, based on workload", + "description": "Per action limit of concurrent manifest data restores, based on workload (ex. Namespace, Deployment, StatefulSet, VirtualMachine)" + }, + "csiSnapshotRestoresPerAction": { + "type": "integer", + "default": 3, + "title": "Per action limit of concurrent CSI volume provisioning requests", + "description": "Per action limit of concurrent CSI volume provisioning requests when restoring from VolumeSnapshots" + }, + "volumeRestoresPerAction": { + "type": "integer", + "default": 3, + "title": "Per action limit of concurrent volume restore operations", + "description": "Per action limit of concurrent volume restore operations from an exported backup" + }, + "workloadSnapshotsPerAction": { + "type": "integer", + "default": 5, + "title": "Per action limit of concurrent manifest data snapshots, based on workload", + "description": "Per action limit of concurrent manifest data snapshots, based on workload (ex. Namespace, Deployment, StatefulSet, VirtualMachine)" + } + } + }, + "gateway": { + "type": "object", + "title": "Gateway config", + "description": "Configure Gateway service", + "properties": { + "insecureDisableSSLVerify": { + "type": "boolean", + "default": false, + "title": "Disable SSL verification for gateway pods", + "description": "Whether to disable SSL verification for gateway pods" + }, + "exposeAdminPort": { + "type": "boolean", + "default": true, + "title": "Expose Admin port", + "description": "Whether to expose Admin port for gateway service" + }, + "service": { + "type": "object", + "title": "gateway service config", + "properties": { + "externalPort": { + "type": "integer", + "default": 80, + "title": "externalPort for the gateway service", + "description": "Override default 80 externalPort for the gateway service" + } + } + }, + "resources": { + "type": "object", + "title": "Gateway pod resource config", + "description": "Configure resource request and limits by Gateway pod", + "properties": { + "requests": { + "type": "object", + "title": "Gateway resource requests", + "description": "Gateway resource requests configuration", + "properties": { + "memory": { + "type": "string", + "default": "300Mi", + "title": "Gateway pod memory request", + "description": "Gateway pod memory request", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "200m", + "title": "Gateway pod cpu request", + "description": "Gateway pod cpu request", + "examples": [ + "1" + ] + } + } + }, + "limits": { + "type": "object", + "title": "Gateway resource limits", + "description": "Gateway resource limits configuration", + "properties": { + "memory": { + "type": "string", + "default": "1Gi", + "title": "Gateway pod memory limit", + "description": "Gateway pod memory limit", + "examples": [ + "1Gi" + ] + }, + "cpu": { + "type": "string", + "default": "1000m", + "title": "Gateway pod cpu limit", + "description": "Gateway pod cpu limit", + "examples": [ + "1" + ] + } + } + } + } + } + } + }, + "kanister": { + "type": "object", + "title": "Kanister config", + "description": "Configuration for Kanister service", + "properties": { + "backupTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout on Kanister backup operations", + "description": "Deprecated. Use 'timeout.blueprintBackup' parameter instead" + }, + "restoreTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister restore operations", + "description": "Deprecated. Use 'timeout.blueprintRestore' parameter instead" + }, + "deleteTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister delete operations", + "description": "Deprecated. Use 'timeout.blueprintDelete' parameter instead" + }, + "hookTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister pre-hook and post-hook operations", + "description": "Deprecated. Use 'timeout.blueprintHooks' parameter instead" + }, + "checkRepoTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister checkRepo operations", + "description": "Deprecated. Use 'timeout.checkRepoPodReady' parameter instead" + }, + "statsTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister stats operations", + "description": "Deprecated. Use 'timeout.statsPodReady' parameter instead" + }, + "efsPostRestoreTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister efsPostRestore operations", + "description": "Deprecated. Use 'timeout.efsRestorePodReady' parameter instead" + }, + "podReadyWaitTimeout": { + "type": "integer", + "default": -1, + "deprecated": true, + "title": "Deprecated: Timeout for Kanister tooling pods to be ready", + "description": "Deprecated. Use 'timeout.workerPodReady' parameter instead" + }, + "managedDataServicesBlueprintsEnabled": { + "type": "boolean", + "default": true, + "title": "Enable built-in Kanister Blueprints for data services", + "description": "Whether to enable built-in Kanister Blueprints for data services such as Crunchy Data Postgres Operator and K8ssandra" + } + } + }, + "timeout": { + "type": "object", + "title": "Timeouts config", + "description": "Configuration of timeouts", + "properties": { + "blueprintBackup": { + "type": "integer", + "default": 45, + "title": "Specifies the timeout (in minutes) for Blueprint backup actions", + "description": "Specifies the timeout (in minutes) for Blueprint backup actions" + }, + "blueprintRestore": { + "type": "integer", + "default": 600, + "title": "Specifies the timeout (in minutes) for Blueprint restore actions", + "description": "Specifies the timeout (in minutes) for Blueprint restore actions" + }, + "blueprintDelete": { + "type": "integer", + "default": 45, + "title": "Specifies the timeout (in minutes) for Blueprint delete actions", + "description": "Specifies the timeout (in minutes) for Blueprint delete actions" + }, + "blueprintHooks": { + "type": "integer", + "default": 20, + "title": "Specifies the timeout (in minutes) for Blueprint backupPrehook and backupPosthook actions", + "description": "Specifies the timeout (in minutes) for Blueprint backupPrehook and backupPosthook actions" + }, + "checkRepoPodReady": { + "type": "integer", + "default": 20, + "title": "Specifies the timeout (in minutes) for temporary worker Pods used to validate backup repository existence", + "description": "Specifies the timeout (in minutes) for temporary worker Pods used to validate backup repository existence" + }, + "statsPodReady": { + "type": "integer", + "default": 20, + "title": "Specifies the timeout (in minutes) for temporary worker Pods used to collect repository statistics", + "description": "Specifies the timeout (in minutes) for temporary worker Pods used to collect repository statistics" + }, + "efsRestorePodReady": { + "type": "integer", + "default": 45, + "title": "Specifies the timeout (in minutes) for temporary worker Pods used for shareable volume restore operations", + "description": "Specifies the timeout (in minutes) for temporary worker Pods used for shareable volume restore operations" + }, + "workerPodReady": { + "type": "integer", + "default": 15, + "title": "Specifies the timeout (in minutes) for all other temporary worker Pods used during Veeam Kasten review comments fixedoperations", + "description": "Specifies the timeout (in minutes) for all other temporary worker Pods used during Veeam Kasten operations" + }, + "jobWait": { + "type": "string", + "default": "", + "title": "Specifies the timeout (in minutes) for completing execution of any child job, after which the parent job will be canceled", + "description": "Specifies the timeout (in minutes) for completing execution of any child job, after which the parent job will be canceled. If no value is set, a default of 10 hours will be used" + } + } + }, + "awsConfig": { + "type": "object", + "title": "AWS config", + "description": "AWS config", + "properties": { + "assumeRoleDuration": { + "type": "string", + "default": "", + "title": "Duration of a session token generated by AWS for an IAM role", + "description": "The minimum value is 15 minutes, and the maximum value is determined by the maximum session duration setting for that IAM role. For documentation on how to view and edit the maximum session duration for an IAM role, refer to https://docs.aws.amazon.com/IAM/latest/UserGuide/id_roles_use.html#id_roles_use_view-role-max-session. The value accepts a number followed by a single character, 'm' (for minutes) or 'h' (for hours). Examples include: 60m or 2h" + }, + "efsBackupVaultName": { + "type": "string", + "default": "k10vault", + "title": "the AWS EFS backup vault name", + "description": "Set the AWS EFS backup vault name" + } + } + }, + "azure": { + "type": "object", + "title": "Azure config", + "description": "Azure config", + "properties": { + "useDefaultMSI": { + "type": "boolean", + "default": false, + "title": "Use the default Managed Identity", + "description": "Set to true - profile does not need a secret, Default Managed Identity will be used" + }, + "useFederatedIdentity": { + "type": "boolean", + "default": false, + "title": "Use the Federated Identity", + "description": "Set to true - injected Federated Identity will be used" + } + } + }, + "google": { + "type": "object", + "title": "Google config", + "description": "Google auth config", + "properties": { + "workloadIdentityFederation": { + "type": "object", + "title": "Google Workload Identity Federation config", + "description": "config for Google Workload Identity Federation", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable Google Workload Identity Federation (GWIF) for K10", + "description": "Set to true - Google Workload Identity Federation is enabled for K10" + }, + "idp": { + "type": "object", + "title": "Identity Provider config", + "description": "Identity Provider config", + "properties": { + "type": { + "type": "string", + "default": "", + "title": "Type of the Identity Provider for GWIF", + "description": "Set the type of IdP for GWIF" + }, + "aud": { + "type": "string", + "default": "", + "title": "The audience that ID token is intended for", + "description": "Set the name of the audience that ID token is intended for" + } + } + } + } + } + } + }, + "grafana": { + "type": "object", + "title": "Grafana config", + "description": "Settings for Grafana service", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "title": "Enable Grafana service", + "description": "Deploy Grafana service. If false Grafana will not be available" + }, + "external": { + "type": "object", + "title": "Configuration related to externally installed Grafana instance", + "description": "If Grafana instance that gets installed with K10 is disabled using grafana.enabled=false, this field can be used to configure externally installed Grafana instance.", + "properties": { + "url": { + "type": "string", + "default": "", + "title": "URL of externally installed Grafana instance", + "description": "If Grafana instance that gets installed with K10 is disabled using grafana.enabled=false, this field can be used to specify URL of externally installed Grafana instance." + } + } + } + } + }, + "encryption": { + "type": "object", + "title": "Encryption config", + "description": "Encryption config", + "properties": { + "primaryKey": { + "type": "object", + "title": "primaryKey for encrypting of K10 primary key", + "description": "primaryKey is used for enabling encryption of K10 primary key", + "properties": { + "awsCmkKeyId": { + "type": "string", + "default": "", + "title": "The AWS CMK key ID for encrypting K10 Primary Key", + "description": "Ensures AWS CMK is used for encrypting K10 primary key" + }, + "vaultTransitKeyName": { + "type": "string", + "default": "", + "title": "Vault transit Key Name", + "description": "Vault Transit key name for Vault integration" + }, + "vaultTransitPath": { + "type": "string", + "default": "", + "title": "Vault transit path", + "description": "Vault transit path for Vault integration" + } + } + } + } + }, + "vmWare": { + "type": "object", + "title": "VMWare integration config", + "properties": { + "taskTimeoutMin": { + "type": "integer", + "default": 60, + "title": "the timeout for VMWare operations", + "description": "the timeout for VMWare operations in minutes" + } + } + }, + "vault": { + "type": "object", + "title": "Vault config", + "description": "Vault integration configuration", + "properties": { + "secretName": { + "type": "string", + "default": "", + "title": "Vault secret name", + "description": "Vault secret name" + }, + "address": { + "type": "string", + "default": "http://vault.vault.svc:8200", + "title": "Vault address", + "description": "Specify Vault endpoint" + }, + "role": { + "type": "string", + "default": "", + "title": "Vault Service Account Role", + "description": "Role that was bound to the service account name and namespace from cluster" + }, + "serviceAccountTokenPath": { + "type": "string", + "default": "", + "title": "Token path for Vault Service Account Role", + "description": "Default: '/var/run/secrets/kubernetes.io/serviceaccount/token'" + } + } + }, + "kubeVirtVMs": { + "type": "object", + "properties": { + "snapshot": { + "type": "object", + "properties": { + "unfreezeTimeout": { + "type": "string", + "title": "Unfreeze timeout for Virtual Machines", + "description": "Time within which K10 is expected to complete the Virtual Machine's backup and thaw the Virtual Machine.", + "default": "5m" + } + } + } + } + }, + "excludedApps": { + "type": "array", + "items": { + "type": "string" + }, + "default": [ + "kube-system", + "kube-ingress", + "kube-node-lease", + "kube-public", + "kube-rook-ceph" + ], + "title": "List of applications to be excluded", + "description": "List of applications to be excluded from the dashboard & compliance considerations" + }, + "logging": { + "type": "object", + "properties": { + "internal": { + "title": "Enable internal logging service", + "description": "Enable use of internal logging service", + "type": "boolean", + "default": true + }, + "fluentbit_endpoint": { + "title": "Use external fluentbit endpoint", + "description": "Specify a fluentbit endpoint to collect logs, cannot be used with the internal logging service (logging.internal=true)", + "type": "string", + "default": "" + } + } + }, + "maxJobWaitDuration": { + "type": "string", + "default": "", + "deprecated": true, + "title": "Deprecated: Maximum duration for jobs in minutes", + "description": "Deprecated. Use 'timeout.jobWait' parameter instead" + }, + "forceRootInKanisterHooks": { + "type": "boolean", + "default": false, + "deprecated": true, + "title": "Deprecated: Run Kanister Hooks as root", + "description": "Deprecated. Use 'forceRootInBlueprintActions' parameter instead" + }, + "forceRootInBlueprintActions": { + "type": "boolean", + "default": true, + "title": "Run Kanister Blueprints as root", + "description": "Forces any Pod created by a Blueprint to run as root user" + }, + "ephemeralPVCOverhead": { + "type": "string", + "default": "0.1", + "title": "Storage overhead for ephemeral PVCs", + "description": "Set the percentage increase for the ephemeral Persistent Volume Claim's storage request, e.g. pvc size = (file raw size) * (1 + `ephemeralPVCOverhead`)" + }, + "kastenDisasterRecovery": { + "type": "object", + "properties": { + "quickMode": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "description": "Enables K10 Quick Disaster Recovery feature, with ability to restore necessary K10 resources and exported restore points of applications.", + "title": "Enable K10 Quick Disaster Recovery." + } + } + } + } + }, + "fips": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "description": "Enables K10 FIPS (Federal Information Processing Standard) mode of operation.", + "title": "Enable K10 FIPS mode of operation." + } + } + }, + "workerPodCRDs": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "default": false, + "title": "Enable ActionPodSpec and ActionPodSpecBinding CRDs", + "description": "Enables the use of ActionPodSpec and ActionPodSpecBinding for granular resource configuration of temporary Pods associated with Kasten Actions" + }, + "resourcesRequests": { + "type": "object", + "title": "Maximum resource requests", + "description": "Specifies the cluster-wide, maximum values for resource requests which may be used in any ActionPodSpec", + "properties": { + "maxMemory": { + "type": "string", + "default": "", + "title": "Maximum memory request", + "description": "Specifies the cluster-wide, maximum value for memory resource requests which may be used in any ActionPodSpec", + "examples": [ + "1Gi" + ] + }, + "maxCPU": { + "type": "string", + "default": "", + "title": "Maximum cpu request", + "description": "Specifies the cluster-wide, maximum value for CPU resource requests which may be used in any ActionPodSpec", + "examples": [ + "1" + ] + } + } + }, + "defaultActionPodSpec": { + "type": "string", + "default": "", + "title": "Default ActionPodSpec name", + "description": "The name of ActionPodSpec that will be used by default for worker pod resources. if empty, the default APS is omitted" + } + } + } + } +} diff --git a/charts/kasten/k10/7.0.1401/values.yaml b/charts/kasten/k10/7.0.1401/values.yaml new file mode 100644 index 000000000..2a20ddb80 --- /dev/null +++ b/charts/kasten/k10/7.0.1401/values.yaml @@ -0,0 +1,606 @@ +#file: noinspection ComposeUnknownKeys,ComposeUnknownValues +# Default values for k10. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +rbac: + 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 derived using the release and chart names. + name: "" + +scc: + create: false + priority: 0 + +networkPolicy: + create: true + +global: + # These are the default values for picking k10 images. They can be overridden + # to specify a particular registy and tag. + image: + registry: gcr.io/kasten-images + tag: '' + pullPolicy: Always + airgapped: + repository: '' + persistence: + mountPath: "/mnt/k10state" + enabled: true + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + storageClass: "" + accessMode: ReadWriteOnce + size: 20Gi + metering: + size: 2Gi + catalog: + size: "" + jobs: + size: "" + logging: + size: "" + grafana: + # Default value is set to 5Gi. This is the same as the default value + # from previous releases <= 4.5.1 where the Grafana sub chart used to + # reference grafana.persistence.size instead of the global values. + # Since the size remains the same across upgrades, the Grafana PVC + # is not deleted and recreated which means no Grafana data is lost + # while upgrading from <= 4.5.1 + size: 5Gi + podLabels: {} + podAnnotations: {} + ## Set it to true while generating helm operator + rhMarketPlace: false + ## these values should not be provided us, these are to be used by + ## red hat marketplace + images: + aggregatedapis: '' + auth: '' + bloblifecyclemanager: '' + catalog: '' + configmap-reload: '' + controllermanager: '' + crypto: '' + dashboardbff: '' + datamover: '' + dex: '' + emissary: '' + events: '' + executor: '' + frontend: '' + gateway: '' + grafana: '' + init: '' + jobs: '' + kanister-tools: '' + kanister: '' + k10tools: '' + logging: '' + metering: '' + ocpconsoleplugin: '' + paygo_daemonset: '' + prometheus: '' + repositories: '' + state: '' + upgrade: '' + vbrintegrationapi: '' + garbagecollector: '' + metric-sidecar: '' + imagePullSecret: '' + prometheus: + external: + host: '' #FQDN of prometheus-service + port: '' + baseURL: '' + network: + enable_ipv6: false + +## OpenShift route configuration. +route: + enabled: false + # Host name for the route + host: "" + # Default path for the route + path: "" + + annotations: {} + # kubernetes.io/tls-acme: "true" + # haproxy.router.openshift.io/disable_cookies: "true" + # haproxy.router.openshift.io/balance: roundrobin + + labels: {} + # key: value + + # TLS configuration + tls: + enabled: false + # What to do in case of an insecure traffic edge termination + insecureEdgeTerminationPolicy: "Redirect" + # Where this TLS configuration should terminate + termination: "edge" + +dexImage: + registry: ghcr.io + repository: dexidp + image: dex + +kanisterToolsImage: + registry: ghcr.io + repository: kanisterio + image: kanister-tools + pullPolicy: Always + +ingress: + create: false + annotations: {} + name: "" + tls: + enabled: false + secretName: "" #TLS secret name + class: "" #Ingress controller type + host: "" #ingress object host name + urlPath: "" #url path for k10 gateway + pathType: "ImplementationSpecific" + defaultBackend: + service: + enabled: false + name: "" + port: + name: "" + number: 0 + resource: + enabled: false + apiGroup: "" + kind: "" + name: "" + +eula: + accept: false #true value if EULA accepted + +license: "" #base64 encoded string provided by Kasten + +cluster: + domainName: "" + +multicluster: + enabled: true + primary: + create: false + name: "" + ingressURL: "" + +prometheus: + rbac: + create: false + server: + # UID and groupid are from prometheus helm chart + enabled: true + securityContext: + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + fsGroup: 65534 + retention: 30d + persistentVolume: + storageClass: "" + fullnameOverride: prometheus-server + baseURL: /k10/prometheus/ + prefixURL: /k10/prometheus + +jaeger: + enabled: false + agentDNS: "" + +service: + externalPort: 8000 + internalPort: 8000 + aggregatedApiPort: 10250 + gatewayAdminPort: 8877 + +secrets: + awsAccessKeyId: '' + awsSecretAccessKey: '' + awsIamRole: '' + awsClientSecretName: '' + googleApiKey: '' + googleProjectId: '' + googleClientSecretName: '' + dockerConfig: '' + dockerConfigPath: '' + azureTenantId: '' + azureClientId: '' + azureClientSecret: '' + azureClientSecretName: '' + azureResourceGroup: '' + azureSubscriptionID: '' + azureResourceMgrEndpoint: '' + azureADEndpoint: '' + azureADResourceID: '' + microsoftEntraIDEndpoint: '' + microsoftEntraIDResourceID: '' + azureCloudEnvID: '' + apiTlsCrt: '' + apiTlsKey: '' + tlsSecret: '' + vsphereEndpoint: '' + vsphereUsername: '' + vspherePassword: '' + vsphereClientSecretName: '' + +metering: + reportingKey: "" #[base64-encoded key] + consumerId: "" #project: + awsRegion: '' + awsMarketPlaceIamRole: '' + awsMarketplace: false # AWS cloud metering license mode + awsManagedLicense: false # AWS managed license mode + licenseConfigSecretName: '' # AWS managed license config secret for non-eks clusters + serviceAccount: + create: false + name: "" + mode: '' # controls metric and license reporting (set to `airgap` for private-network installs) + redhatMarketplacePayg: false # Redhat cloud metering license mode + reportCollectionPeriod: 1800 # metric report collection period in seconds + reportPushPeriod: 3600 # metric report push period in seconds + promoID: '' # sets the K10 promotion ID + +clusterName: '' + +# Deprecated. Use 'limiter.executorReplicas' parameter. +executorReplicas: -1 + +logLevel: info + +externalGateway: + create: false + # Any standard service annotations + annotations: {} + # Host and domain name for the K10 API server + fqdn: + name: "" + #Supported types route53-mapper, external-dns + type: "" + # ARN for the AWS ACM SSL certificate used in the K10 API server (load balancer) + awsSSLCertARN: '' + +auth: + groupAllowList: [] +# - "group1" +# - "group2" + basicAuth: + enabled: false + secretName: "" #htpasswd based existing secret + htpasswd: "" #htpasswd string, which will be used for basic auth + tokenAuth: + enabled: false + oidcAuth: + enabled: false + providerURL: "" #URL to your OIDC provider + redirectURL: "" #URL to the K10 gateway service + scopes: "" #Space separated OIDC scopes required for userinfo. Example: "profile email" + prompt: "select_account" #The prompt type to be requested with the OIDC provider. Default is select_account. + clientID: "" #ClientID given by the OIDC provider for K10 + clientSecret: "" #ClientSecret given by the OIDC provider for K10 + clientSecretName: "" #The Kubernetes Secret that contains ClientID and ClientSecret given by the OIDC provider for K10 + usernameClaim: "" #Claim to be used as the username + usernamePrefix: "" #Prefix that has to be used with the username obtained from the username claim + groupClaim: "" #Name of a custom OpenID Connect claim for specifying user groups + groupPrefix: "" #All groups will be prefixed with this value to prevent conflicts. + logoutURL: "" #URL to your OIDC provider's logout endpoint + #OIDC config based existing secret. + #Must include providerURL, redirectURL, scopes, clientID/secret and logoutURL. + secretName: "" + sessionDuration: "1h" #Maximum OIDC session duration. Default value is 1 hour + refreshTokenSupport: false #Enable Refresh Token support. Disabled by default + openshift: + enabled: false + serviceAccount: "" #service account used as the OAuth client + clientSecret: "" #The token from the service account + clientSecretName: "" #The secret with the token from the service account + dashboardURL: "" #The URL for accessing K10's dashboard + openshiftURL: "" #The URL of the Openshift API server + insecureCA: false + useServiceAccountCA: false + secretName: "" # The Kubernetes Secret that contains OIDC settings + usernameClaim: "email" + usernamePrefix: "" + groupnameClaim: "groups" + groupnamePrefix: "" + caCertsAutoExtraction: true # Configures if K10 should automatically extract CA certificates from the OCP cluster. + ldap: + enabled: false + restartPod: false # Enable this value to force a restart of the authentication service pod + dashboardURL: "" #The URL for accessing K10's dashboard + host: "" + insecureNoSSL: false + insecureSkipVerifySSL: false + startTLS: false + bindDN: "" + bindPW: "" + bindPWSecretName: "" + userSearch: + baseDN: "" + filter: "" + username: "" + idAttr: "" + emailAttr: "" + nameAttr: "" + preferredUsernameAttr: "" + groupSearch: + baseDN: "" + filter: "" + userMatchers: [] +# - userAttr: +# groupAttr: + nameAttr: "" + secretName: "" # The Kubernetes Secret that contains OIDC settings + usernameClaim: "email" + usernamePrefix: "" + groupnameClaim: "groups" + groupnamePrefix: "" + k10AdminUsers: [] + k10AdminGroups: [] + +optionalColocatedServices: + vbrintegrationapi: + enabled: true + +cacertconfigmap: + name: "" #Name of the configmap + +apiservices: + deployed: true # If false APIService objects will not be deployed + +injectKanisterSidecar: + enabled: false + namespaceSelector: + matchLabels: {} + # Set objectSelector to filter workloads + objectSelector: + matchLabels: {} + webhookServer: + port: 8080 # should not conflict with config server port (8000) + +genericStorageBackup: + token: "" + +kanisterPodCustomLabels : "" + +kanisterPodCustomAnnotations : "" + +features: + backgroundMaintenanceRun: true # Key must be deleted to deactivate. Setting to false will not work. + +# Deprecated. Use 'workerPodMetricSidecar' parameter instead. +kanisterPodMetricSidecar: + enabled: false + metricLifetime: "2m" + pushGatewayInterval: "30s" + resources: + requests: + memory: "" + cpu: "" + limits: + memory: "" + cpu: "" + +workerPodMetricSidecar: + enabled: true + metricLifetime: "2m" + pushGatewayInterval: "30s" + resources: + requests: + memory: "" + cpu: "" + limits: + memory: "" + cpu: "" + +genericVolumeSnapshot: + resources: + requests: + memory: "" + cpu: "" + limits: + memory: "" + cpu: "" + +garbagecollector: + daemonPeriod: 21600 + keepMaxActions: 1000 + actions: + enabled: false + +resources: {} + +defaultPriorityClassName: "" +priorityClassName: {} + +services: + executor: + hostNetwork: false + # Deprecated. Use 'limiter.executorThreads' parameter instead. + workerCount: -1 + # Deprecated. Use 'limiter.csiSnapshotRestoresPerAction' parameter instead. + maxConcurrentRestoreCsiSnapshots: -1 + # Deprecated. Use 'limiter.volumeRestoresPerAction' parameter instead. + maxConcurrentRestoreGenericVolumeSnapshots: -1 + # Deprecated. Use 'limiter.workloadRestoresPerAction' parameter instead. + maxConcurrentRestoreWorkloads: -1 + dashboardbff: + hostNetwork: false + securityContext: + runAsUser: 1000 # Will override any USER instruction that a container image set for running the entrypoint and command. + fsGroup: 1000 + runAsNonRoot: true + seccompProfile: + type: RuntimeDefault + aggregatedapis: + hostNetwork: false + +siem: + logging: + cluster: + enabled: true + cloud: + path: k10audit/ + awsS3: + enabled: true + +apigateway: + serviceResolver: dns + +limiter: + # Deprecated. Use 'limiter.snapshotExportsPerAction' parameter instead. + concurrentSnapConversions: -1 + snapshotExportsPerAction: 3 + # Deprecated. Use 'limiter.genericVolumeBackupsPerCluster' parameter instead. + genericVolumeSnapshots: -1 + genericVolumeBackupsPerCluster: 10 + # Deprecated. Use 'limiter.snapshotExportsPerCluster' parameter instead. + genericVolumeCopies: -1 + snapshotExportsPerCluster: 10 + # Deprecated. Use 'limiter.volumeRestoresPerCluster' parameter instead. + genericVolumeRestores: -1 + volumeRestoresPerCluster: 10 + # Deprecated. Use 'limiter.csiSnapshotsPerCluster' parameter instead. + csiSnapshots: -1 + csiSnapshotsPerCluster: 10 + # Deprecated. Use 'limiter.directSnapshotsPerCluster' parameter instead. + providerSnapshots: -1 + directSnapshotsPerCluster: 10 + # Deprecated. Use 'limiter.imageCopiesPerCluster' parameter instead. + imageCopies: -1 + imageCopiesPerCluster: 10 + executorReplicas: 3 + executorThreads: 8 + workloadRestoresPerAction: 3 + csiSnapshotRestoresPerAction: 3 + volumeRestoresPerAction: 3 + workloadSnapshotsPerAction: 5 + +gateway: + insecureDisableSSLVerify: false + exposeAdminPort: true + service: + externalPort: 80 + resources: + requests: + memory: 300Mi + cpu: 200m + limits: + memory: 1Gi + cpu: 1000m + +kanister: + # Deprecated. Use 'timeout.blueprintBackup' parameter instead. + backupTimeout: -1 + # Deprecated. Use 'timeout.blueprintRestore' parameter instead. + restoreTimeout: -1 + # Deprecated. Use 'timeout.blueprintDelete' parameter instead. + deleteTimeout: -1 + # Deprecated. Use 'timeout.blueprintHooks' parameter instead. + hookTimeout: -1 + # Deprecated. Use 'timeout.checkRepoPodReady' parameter instead. + checkRepoTimeout: -1 + # Deprecated. Use 'timeout.statsPodReady' parameter instead. + statsTimeout: -1 + # Deprecated. Use 'timeout.efsRestorePodReady' parameter instead. + efsPostRestoreTimeout: -1 + # Deprecated. Use 'timeout.workerPodReady' parameter instead. + podReadyWaitTimeout: -1 + managedDataServicesBlueprintsEnabled: true + +timeout: + blueprintBackup: 45 + blueprintRestore: 600 + blueprintDelete: 45 + blueprintHooks: 20 + checkRepoPodReady: 20 + statsPodReady: 20 + efsRestorePodReady: 45 + workerPodReady: 15 + jobWait: "" + +awsConfig: + assumeRoleDuration: "" + efsBackupVaultName: "k10vault" + +excludedApps: ["kube-system", "kube-ingress", "kube-node-lease", "kube-public", "kube-rook-ceph"] + +grafana: + enabled: true + external: + url: "" # can be used to configure the URL of externally installed Grafana instance. If it's provided, grafana.enabled must be false. + +encryption: + primaryKey: # primaryKey is used for enabling encryption of K10 primary key + awsCmkKeyId: '' # Ensures AWS CMK is used for encrypting K10 primary key + vaultTransitKeyName: '' + vaultTransitPath: '' + +vmWare: + taskTimeoutMin: 60 + +azure: + useDefaultMSI: false + useFederatedIdentity: false + +google: + workloadIdentityFederation: + enabled: false + idp: + type: "" + aud: "" + +vault: + role: "" # Role that was bound to the service account name and namespace from cluster + serviceAccountTokenPath: "" # This will default to /var/run/secrets/kubernetes.io/serviceaccount/token within the code if left blank + address: "http://vault.vault.svc:8200" # Address for dev mode in cluster vault server in vault namespace + secretName: "" # Ensures backward compatibility for now. We can remove once we tell all customers this is deprecated. + # This is how the token can be passed into default if K8S auth mode fails for whatever reason. + +kubeVirtVMs: + snapshot: + unfreezeTimeout: "5m" + +logging: + internal: true + # Used to set an external fluentbit endpoint. 'logging.internal' must be set to false. + fluentbit_endpoint: "" + +# Deprecated. Use 'timeout.jobWait' parameter instead. +maxJobWaitDuration: "" + +# Deprecated. Use 'forceRootInBlueprintActions' parameter instead. +forceRootInKanisterHooks: false +forceRootInBlueprintActions: true + +ephemeralPVCOverhead: "0.1" + +datastore: + parallelUploads: 8 + parallelDownloads: 8 + +kastenDisasterRecovery: + quickMode: + enabled: false + +fips: + enabled: false + +workerPodCRDs: + enabled: false + resourcesRequests: + maxCPU: "" + maxMemory: "" + defaultActionPodSpec: "" diff --git a/charts/kuma/kuma/2.9.1/.helmdocsignore b/charts/kuma/kuma/2.9.1/.helmdocsignore new file mode 100644 index 000000000..d8a5db8f8 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/.helmdocsignore @@ -0,0 +1 @@ +# Charts to ignore from helm-docs \ No newline at end of file diff --git a/charts/kuma/kuma/2.9.1/.helmignore b/charts/kuma/kuma/2.9.1/.helmignore new file mode 100644 index 000000000..0e8a0eb36 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/.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 +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/kuma/kuma/2.9.1/Chart.yaml b/charts/kuma/kuma/2.9.1/Chart.yaml new file mode 100644 index 000000000..8fdc14f2c --- /dev/null +++ b/charts/kuma/kuma/2.9.1/Chart.yaml @@ -0,0 +1,26 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Kuma + catalog.cattle.io/namespace: kuma-system + catalog.cattle.io/release-name: kuma +apiVersion: v2 +appVersion: 2.9.1 +description: A Helm chart for the Kuma Control Plane +home: https://github.com/kumahq/kuma +icon: file://assets/icons/kuma.svg +keywords: +- service mesh +- control plane +maintainers: +- email: jakub.dyszkiewicz@konghq.com + name: Jakub Dyszkiewicz + url: https://github.com/jakubdyszkiewicz +- email: charly.molter@konghq.com + name: Charly Molter + url: https://github.com/lahabana +- email: michael.beaumont@konghq.com + name: Mike Beaumont + url: https://github.com/michaelbeaumont +name: kuma +type: application +version: 2.9.1 diff --git a/charts/kuma/kuma/2.9.1/README.md b/charts/kuma/kuma/2.9.1/README.md new file mode 100644 index 000000000..e9dcc2b22 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/README.md @@ -0,0 +1,316 @@ +[![][kuma-logo]][kuma-url] + +A Helm chart for the Kuma Control Plane + +![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![Version: 2.9.1](https://img.shields.io/badge/Version-2.9.1-informational?style=flat-square) ![AppVersion: 2.9.1](https://img.shields.io/badge/AppVersion-2.9.1-informational?style=flat-square) + +**Homepage:** + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| global.image.registry | string | `"docker.io/kumahq"` | Default registry for all Kuma Images | +| global.image.tag | string | `nil` | The default tag for all Kuma images, which itself defaults to .Chart.AppVersion | +| global.imagePullSecrets | list | `[]` | Add `imagePullSecrets` to all the service accounts used for Kuma components | +| patchSystemNamespace | bool | `true` | Whether to patch the target namespace with the system label | +| installCrdsOnUpgrade.enabled | bool | `true` | Whether install new CRDs before upgrade (if any were introduced with the new version of Kuma) | +| installCrdsOnUpgrade.imagePullSecrets | list | `[]` | The `imagePullSecrets` to attach to the Service Account running CRD installation. This field will be deprecated in a future release, please use .global.imagePullSecrets | +| noHelmHooks | bool | `false` | Whether to disable all helm hooks | +| restartOnSecretChange | bool | `true` | Whether to restart control-plane by calculating a new checksum for the secret | +| controlPlane.environment | string | `"kubernetes"` | Environment that control plane is run in, useful when running universal global control plane on k8s | +| controlPlane.extraLabels | object | `{}` | Labels to add to resources in addition to default labels | +| controlPlane.logLevel | string | `"info"` | Kuma CP log level: one of off,info,debug | +| controlPlane.logOutputPath | string | `""` | Kuma CP log output path: Defaults to /dev/stdout | +| controlPlane.mode | string | `"zone"` | Kuma CP modes: one of zone,global | +| controlPlane.zone | string | `nil` | Kuma CP zone, if running multizone | +| controlPlane.kdsGlobalAddress | string | `""` | Only used in `zone` mode | +| controlPlane.replicas | int | `1` | Number of replicas of the Kuma CP. Ignored when autoscaling is enabled | +| controlPlane.minReadySeconds | int | `0` | Minimum number of seconds for which a newly created pod should be ready for it to be considered available. | +| controlPlane.deploymentAnnotations | object | `{}` | Annotations applied only to the `Deployment` resource | +| controlPlane.podAnnotations | object | `{}` | Annotations applied only to the `Pod` resource | +| controlPlane.autoscaling.enabled | bool | `false` | Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster | +| controlPlane.autoscaling.minReplicas | int | `2` | The minimum CP pods to allow | +| controlPlane.autoscaling.maxReplicas | int | `5` | The max CP pods to scale to | +| controlPlane.autoscaling.targetCPUUtilizationPercentage | int | `80` | For clusters that don't support autoscaling/v2, autoscaling/v1 is used | +| controlPlane.autoscaling.metrics | list | `[{"resource":{"name":"cpu","target":{"averageUtilization":80,"type":"Utilization"}},"type":"Resource"}]` | For clusters that do support autoscaling/v2, use metrics | +| controlPlane.nodeSelector | object | `{"kubernetes.io/os":"linux"}` | Node selector for the Kuma Control Plane pods | +| controlPlane.tolerations | list | `[]` | Tolerations for the Kuma Control Plane pods | +| controlPlane.podDisruptionBudget.enabled | bool | `false` | Whether to create a pod disruption budget | +| controlPlane.podDisruptionBudget.maxUnavailable | int | `1` | The maximum number of unavailable pods allowed by the budget | +| controlPlane.affinity | object | `{"podAntiAffinity":{"preferredDuringSchedulingIgnoredDuringExecution":[{"podAffinityTerm":{"labelSelector":{"matchExpressions":[{"key":"app.kubernetes.io/name","operator":"In","values":["{{ include \"kuma.name\" . }}"]},{"key":"app.kubernetes.io/instance","operator":"In","values":["{{ .Release.Name }}"]},{"key":"app","operator":"In","values":["{{ include \"kuma.name\" . }}-control-plane"]}]},"topologyKey":"kubernetes.io/hostname"},"weight":100}]}}` | Affinity placement rule for the Kuma Control Plane pods. This is rendered as a template, so you can reference other helm variables or includes. | +| controlPlane.topologySpreadConstraints | string | `nil` | Topology spread constraints rule for the Kuma Control Plane pods. This is rendered as a template, so you can use variables to generate match labels. | +| controlPlane.injectorFailurePolicy | string | `"Fail"` | Failure policy of the mutating webhook implemented by the Kuma Injector component | +| controlPlane.service.apiServer.http.nodePort | int | `30681` | Port on which Http api server Service is exposed on Node for service of type NodePort | +| controlPlane.service.apiServer.https.nodePort | int | `30682` | Port on which Https api server Service is exposed on Node for service of type NodePort | +| controlPlane.service.enabled | bool | `true` | Whether to create a service resource. | +| controlPlane.service.name | string | `nil` | Optionally override of the Kuma Control Plane Service's name | +| controlPlane.service.type | string | `"ClusterIP"` | Service type of the Kuma Control Plane | +| controlPlane.service.annotations | object | `{"prometheus.io/port":"5680","prometheus.io/scrape":"true"}` | Annotations to put on the Kuma Control Plane | +| controlPlane.ingress.enabled | bool | `false` | Install K8s Ingress resource that exposes GUI and API | +| controlPlane.ingress.ingressClassName | string | `nil` | IngressClass defines which controller will implement the resource | +| controlPlane.ingress.hostname | string | `nil` | Ingress hostname | +| controlPlane.ingress.annotations | object | `{}` | Map of ingress annotations. | +| controlPlane.ingress.path | string | `"/"` | Ingress path. | +| controlPlane.ingress.pathType | string | `"ImplementationSpecific"` | Each path in an Ingress is required to have a corresponding path type. (ImplementationSpecific/Exact/Prefix) | +| controlPlane.ingress.servicePort | int | `5681` | Port from kuma-cp to use to expose API and GUI. Switch to 5682 to expose TLS port | +| controlPlane.globalZoneSyncService.enabled | bool | `true` | Whether to create a k8s service for the global zone sync service. It will only be created when enabled and deploying the global control plane. | +| controlPlane.globalZoneSyncService.type | string | `"LoadBalancer"` | Service type of the Global-zone sync | +| controlPlane.globalZoneSyncService.loadBalancerIP | string | `nil` | Optionally specify IP to be used by cloud provider when configuring load balancer | +| controlPlane.globalZoneSyncService.loadBalancerSourceRanges | list | `[]` | Optionally specify allowed source ranges that can access the load balancer | +| controlPlane.globalZoneSyncService.annotations | object | `{}` | Additional annotations to put on the Global Zone Sync Service | +| controlPlane.globalZoneSyncService.nodePort | int | `30685` | Port on which Global Zone Sync Service is exposed on Node for service of type NodePort | +| controlPlane.globalZoneSyncService.port | int | `5685` | Port on which Global Zone Sync Service is exposed | +| controlPlane.globalZoneSyncService.protocol | string | `"grpc"` | Protocol of the Global Zone Sync service port | +| controlPlane.defaults.skipMeshCreation | bool | `false` | Whether to skip creating the default Mesh | +| controlPlane.automountServiceAccountToken | bool | `true` | Whether to automountServiceAccountToken for cp. Optionally set to false | +| controlPlane.resources | object | `{"limits":{"memory":"256Mi"},"requests":{"cpu":"500m","memory":"256Mi"}}` | Optionally override the resource spec | +| controlPlane.lifecycle | object | `{}` | Pod lifecycle settings (useful for adding a preStop hook, when using AWS ALB or NLB) | +| controlPlane.terminationGracePeriodSeconds | int | `30` | Number of seconds to wait before force killing the pod. Make sure to update this if you add a preStop hook. | +| controlPlane.tls.general.secretName | string | `""` | Secret that contains tls.crt, tls.key [and ca.crt when no controlPlane.tls.general.caSecretName specified] for protecting Kuma in-cluster communication | +| controlPlane.tls.general.caSecretName | string | `""` | Secret that contains ca.crt that was used to sign cert for protecting Kuma in-cluster communication (ca.crt present in this secret have precedence over the one provided in the controlPlane.tls.general.secretName) | +| controlPlane.tls.general.caBundle | string | `""` | Base64 encoded CA certificate (the same as in controlPlane.tls.general.secret#ca.crt) | +| controlPlane.tls.apiServer.secretName | string | `""` | Secret that contains tls.crt, tls.key for protecting Kuma API on HTTPS | +| controlPlane.tls.apiServer.clientCertsSecretName | string | `""` | Secret that contains list of .pem certificates that can access admin endpoints of Kuma API on HTTPS | +| controlPlane.tls.kdsGlobalServer.secretName | string | `""` | Name of the K8s TLS Secret resource. If you set this and don't set create=true, you have to create the secret manually. | +| controlPlane.tls.kdsGlobalServer.create | bool | `false` | Whether to create the TLS secret in helm. | +| controlPlane.tls.kdsGlobalServer.cert | string | `""` | The TLS certificate to offer. | +| controlPlane.tls.kdsGlobalServer.key | string | `""` | The TLS key to use. | +| controlPlane.tls.kdsZoneClient.secretName | string | `""` | Name of the K8s Secret resource that contains ca.crt which was used to sign the certificate of KDS Global Server. If you set this and don't set create=true, you have to create the secret manually. | +| controlPlane.tls.kdsZoneClient.create | bool | `false` | Whether to create the TLS secret in helm. | +| controlPlane.tls.kdsZoneClient.cert | string | `""` | CA bundle that was used to sign the certificate of KDS Global Server. | +| controlPlane.tls.kdsZoneClient.skipVerify | bool | `false` | If true, TLS cert of the server is not verified. | +| controlPlane.serviceAccountAnnotations | object | `{}` | Annotations to add for Control Plane's Service Account | +| controlPlane.image.pullPolicy | string | `"IfNotPresent"` | Kuma CP ImagePullPolicy | +| controlPlane.image.repository | string | `"kuma-cp"` | Kuma CP image repository | +| controlPlane.image.tag | string | `nil` | Kuma CP Image tag. When not specified, the value is copied from global.tag | +| controlPlane.secrets | object with { Env: string, Secret: string, Key: string } | `nil` | Secrets to add as environment variables, where `Env` is the name of the env variable, `Secret` is the name of the Secret, and `Key` is the key of the Secret value to use | +| controlPlane.envVars | object | `{}` | Additional environment variables that will be passed to the control plane | +| controlPlane.envVarEntries | string | `nil` | Additional environment variables that will be passed to the control plane. Can be used with Kubernetes downward API | +| controlPlane.extraConfigMaps | list | `[]` | Additional config maps to mount into the control plane, with optional inline values | +| controlPlane.extraSecrets | object with { name: string, mountPath: string, readOnly: string } | `nil` | Additional secrets to mount into the control plane, where `Env` is the name of the env variable, `Secret` is the name of the Secret, and `Key` is the key of the Secret value to use | +| controlPlane.webhooks.validator.additionalRules | string | `""` | Additional rules to apply on Kuma validator webhook. Useful when building custom policy on top of Kuma. | +| controlPlane.webhooks.ownerReference.additionalRules | string | `""` | Additional rules to apply on Kuma owner reference webhook. Useful when building custom policy on top of Kuma. | +| controlPlane.hostNetwork | bool | `false` | Specifies if the deployment should be started in hostNetwork mode. | +| controlPlane.admissionServerPort | int | `5443` | Define a new server port for the admission controller. Recommended to set in combination with hostNetwork to prevent multiple port bindings on the same port (like Calico in AWS EKS). | +| controlPlane.podSecurityContext | object | `{"runAsNonRoot":true}` | Security context at the pod level for control plane. | +| controlPlane.containerSecurityContext | object | `{"readOnlyRootFilesystem":true}` | Security context at the container level for control plane. | +| controlPlane.supportGatewaySecretsInAllNamespaces | bool | `false` | If true, then control plane can support TLS secrets for builtin gateway outside of mesh system namespace. The downside is that control plane requires permission to read Secrets in all namespaces. | +| controlPlane.dns | object | `{"config":{"nameservers":[],"searches":[]},"policy":""}` | DNS configuration for the control-plane pod. This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). | +| controlPlane.dns.policy | string | `""` | Defines how DNS resolution is configured for that Pod. | +| controlPlane.dns.config | object | `{"nameservers":[],"searches":[]}` | Optional dns configuration, required when policy is 'None' | +| controlPlane.dns.config.nameservers | list | `[]` | A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. | +| controlPlane.dns.config.searches | list | `[]` | A list of DNS search domains for hostname lookup in the Pod. | +| cni.enabled | bool | `false` | Install Kuma with CNI instead of proxy init container | +| cni.chained | bool | `false` | Install CNI in chained mode | +| cni.netDir | string | `"/etc/cni/multus/net.d"` | Set the CNI install directory | +| cni.binDir | string | `"/var/lib/cni/bin"` | Set the CNI bin directory | +| cni.confName | string | `"kuma-cni.conf"` | Set the CNI configuration name | +| cni.logLevel | string | `"info"` | CNI log level: one of off,info,debug | +| cni.nodeSelector | object | `{"kubernetes.io/os":"linux"}` | Node Selector for the CNI pods | +| cni.tolerations | list | `[]` | Tolerations for the CNI pods | +| cni.podAnnotations | object | `{}` | Additional pod annotations | +| cni.namespace | string | `"kube-system"` | Set the CNI namespace | +| cni.image.repository | string | `"kuma-cni"` | CNI image repository | +| cni.image.tag | string | `nil` | CNI image tag - defaults to .Chart.AppVersion | +| cni.image.imagePullPolicy | string | `"IfNotPresent"` | CNI image pull policy | +| cni.delayStartupSeconds | int | `0` | it's only useful in tests to trigger a possible race condition | +| cni.experimental | object | `{"imageEbpf":{"registry":"docker.io/kumahq","repository":"merbridge","tag":"0.8.5"}}` | use new CNI (experimental) | +| cni.experimental.imageEbpf.registry | string | `"docker.io/kumahq"` | CNI experimental eBPF image registry | +| cni.experimental.imageEbpf.repository | string | `"merbridge"` | CNI experimental eBPF image repository | +| cni.experimental.imageEbpf.tag | string | `"0.8.5"` | CNI experimental eBPF image tag | +| cni.resources.requests.cpu | string | `"100m"` | | +| cni.resources.requests.memory | string | `"100Mi"` | | +| cni.resources.limits.memory | string | `"100Mi"` | | +| cni.podSecurityContext | object | `{}` | Security context at the pod level for cni | +| cni.containerSecurityContext | object | `{"readOnlyRootFilesystem":true,"runAsGroup":0,"runAsNonRoot":false,"runAsUser":0}` | Security context at the container level for cni | +| dataPlane.dnsLogging | bool | `false` | If true, then turn on CoreDNS query logging | +| dataPlane.image.repository | string | `"kuma-dp"` | The Kuma DP image repository | +| dataPlane.image.pullPolicy | string | `"IfNotPresent"` | Kuma DP ImagePullPolicy | +| dataPlane.image.tag | string | `nil` | Kuma DP Image Tag. When not specified, the value is copied from global.tag | +| dataPlane.initImage.repository | string | `"kuma-init"` | The Kuma DP init image repository | +| dataPlane.initImage.tag | string | `nil` | Kuma DP init image tag When not specified, the value is copied from global.tag | +| ingress.enabled | bool | `false` | If true, it deploys Ingress for cross cluster communication | +| ingress.extraLabels | object | `{}` | Labels to add to resources, in addition to default labels | +| ingress.drainTime | string | `"30s"` | Time for which old listener will still be active as draining | +| ingress.replicas | int | `1` | Number of replicas of the Ingress. Ignored when autoscaling is enabled. | +| ingress.logLevel | string | `"info"` | Log level for ingress (available values: off|info|debug) | +| ingress.resources | object | `{"limits":{"cpu":"1000m","memory":"512Mi"},"requests":{"cpu":"50m","memory":"64Mi"}}` | Define the resources to allocate to mesh ingress | +| ingress.lifecycle | object | `{}` | Pod lifecycle settings (useful for adding a preStop hook, when using AWS ALB or NLB) | +| ingress.terminationGracePeriodSeconds | int | `40` | Number of seconds to wait before force killing the pod. Make sure to update this if you add a preStop hook. | +| ingress.autoscaling.enabled | bool | `false` | Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster | +| ingress.autoscaling.minReplicas | int | `2` | The minimum CP pods to allow | +| ingress.autoscaling.maxReplicas | int | `5` | The max CP pods to scale to | +| ingress.autoscaling.targetCPUUtilizationPercentage | int | `80` | For clusters that don't support autoscaling/v2, autoscaling/v1 is used | +| ingress.autoscaling.metrics | list | `[{"resource":{"name":"cpu","target":{"averageUtilization":80,"type":"Utilization"}},"type":"Resource"}]` | For clusters that do support autoscaling/v2, use metrics | +| ingress.service.enabled | bool | `true` | Whether to create a Service resource. | +| ingress.service.type | string | `"LoadBalancer"` | Service type of the Ingress | +| ingress.service.loadBalancerIP | string | `nil` | Optionally specify IP to be used by cloud provider when configuring load balancer | +| ingress.service.annotations | object | `{}` | Additional annotations to put on the Ingress service | +| ingress.service.port | int | `10001` | Port on which Ingress is exposed | +| ingress.service.nodePort | string | `nil` | Port on which service is exposed on Node for service of type NodePort | +| ingress.annotations | object | `{}` | Additional pod annotations (deprecated favor `podAnnotations`) | +| ingress.podAnnotations | object | `{}` | Additional pod annotations | +| ingress.nodeSelector | object | `{"kubernetes.io/os":"linux"}` | Node Selector for the Ingress pods | +| ingress.tolerations | list | `[]` | Tolerations for the Ingress pods | +| ingress.podDisruptionBudget.enabled | bool | `false` | Whether to create a pod disruption budget | +| ingress.podDisruptionBudget.maxUnavailable | int | `1` | The maximum number of unavailable pods allowed by the budget | +| ingress.affinity | object | `{"podAntiAffinity":{"preferredDuringSchedulingIgnoredDuringExecution":[{"podAffinityTerm":{"labelSelector":{"matchExpressions":[{"key":"app.kubernetes.io/name","operator":"In","values":["{{ include \"kuma.name\" . }}"]},{"key":"app.kubernetes.io/instance","operator":"In","values":["{{ .Release.Name }}"]},{"key":"app","operator":"In","values":["kuma-ingress"]}]},"topologyKey":"kubernetes.io/hostname"},"weight":100}]}}` | Affinity placement rule for the Kuma Ingress pods This is rendered as a template, so you can reference other helm variables or includes. | +| ingress.topologySpreadConstraints | string | `nil` | Topology spread constraints rule for the Kuma Mesh Ingress pods. This is rendered as a template, so you can use variables to generate match labels. | +| ingress.podSecurityContext | object | `{"runAsGroup":5678,"runAsNonRoot":true,"runAsUser":5678}` | Security context at the pod level for ingress | +| ingress.containerSecurityContext | object | `{"readOnlyRootFilesystem":true}` | Security context at the container level for ingress | +| ingress.serviceAccountAnnotations | object | `{}` | Annotations to add for Control Plane's Service Account | +| ingress.automountServiceAccountToken | bool | `true` | Whether to automountServiceAccountToken for cp. Optionally set to false | +| ingress.dns | object | `{"config":{"nameservers":[],"searches":[]},"policy":""}` | DNS configuration for the ingress pod. This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). | +| ingress.dns.policy | string | `""` | Defines how DNS resolution is configured for that Pod. | +| ingress.dns.config | object | `{"nameservers":[],"searches":[]}` | Optional dns configuration, required when policy is 'None' | +| ingress.dns.config.nameservers | list | `[]` | A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. | +| ingress.dns.config.searches | list | `[]` | A list of DNS search domains for hostname lookup in the Pod. | +| egress.enabled | bool | `false` | If true, it deploys Egress for cross cluster communication | +| egress.extraLabels | object | `{}` | Labels to add to resources, in addition to the default labels. | +| egress.drainTime | string | `"30s"` | Time for which old listener will still be active as draining | +| egress.replicas | int | `1` | Number of replicas of the Egress. Ignored when autoscaling is enabled. | +| egress.logLevel | string | `"info"` | Log level for egress (available values: off|info|debug) | +| egress.autoscaling.enabled | bool | `false` | Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster | +| egress.autoscaling.minReplicas | int | `2` | The minimum CP pods to allow | +| egress.autoscaling.maxReplicas | int | `5` | The max CP pods to scale to | +| egress.autoscaling.targetCPUUtilizationPercentage | int | `80` | For clusters that don't support autoscaling/v2, autoscaling/v1 is used | +| egress.autoscaling.metrics | list | `[{"resource":{"name":"cpu","target":{"averageUtilization":80,"type":"Utilization"}},"type":"Resource"}]` | For clusters that do support autoscaling/v2, use metrics | +| egress.resources.requests.cpu | string | `"50m"` | | +| egress.resources.requests.memory | string | `"64Mi"` | | +| egress.resources.limits.cpu | string | `"1000m"` | | +| egress.resources.limits.memory | string | `"512Mi"` | | +| egress.service.enabled | bool | `true` | Whether to create the service object | +| egress.service.type | string | `"ClusterIP"` | Service type of the Egress | +| egress.service.loadBalancerIP | string | `nil` | Optionally specify IP to be used by cloud provider when configuring load balancer | +| egress.service.annotations | object | `{}` | Additional annotations to put on the Egress service | +| egress.service.port | int | `10002` | Port on which Egress is exposed | +| egress.service.nodePort | string | `nil` | Port on which service is exposed on Node for service of type NodePort | +| egress.annotations | object | `{}` | Additional pod annotations (deprecated favor `podAnnotations`) | +| egress.podAnnotations | object | `{}` | Additional pod annotations | +| egress.nodeSelector | object | `{"kubernetes.io/os":"linux"}` | Node Selector for the Egress pods | +| egress.tolerations | list | `[]` | Tolerations for the Egress pods | +| egress.podDisruptionBudget.enabled | bool | `false` | Whether to create a pod disruption budget | +| egress.podDisruptionBudget.maxUnavailable | int | `1` | The maximum number of unavailable pods allowed by the budget | +| egress.affinity | object | `{"podAntiAffinity":{"preferredDuringSchedulingIgnoredDuringExecution":[{"podAffinityTerm":{"labelSelector":{"matchExpressions":[{"key":"app.kubernetes.io/name","operator":"In","values":["{{ include \"kuma.name\" . }}"]},{"key":"app.kubernetes.io/instance","operator":"In","values":["{{ .Release.Name }}"]},{"key":"app","operator":"In","values":["kuma-egress"]}]},"topologyKey":"kubernetes.io/hostname"},"weight":100}]}}` | Affinity placement rule for the Kuma Egress pods. This is rendered as a template, so you can reference other helm variables or includes. | +| egress.topologySpreadConstraints | string | `nil` | Topology spread constraints rule for the Kuma Egress pods. This is rendered as a template, so you can use variables to generate match labels. | +| egress.podSecurityContext | object | `{"runAsGroup":5678,"runAsNonRoot":true,"runAsUser":5678}` | Security context at the pod level for egress | +| egress.containerSecurityContext | object | `{"readOnlyRootFilesystem":true}` | Security context at the container level for egress | +| egress.serviceAccountAnnotations | object | `{}` | Annotations to add for Control Plane's Service Account | +| egress.automountServiceAccountToken | bool | `true` | Whether to automountServiceAccountToken for cp. Optionally set to false | +| egress.dns | object | `{"config":{"nameservers":[],"searches":[]},"policy":""}` | DNS configuration for the egress pod. This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). | +| egress.dns.policy | string | `""` | Defines how DNS resolution is configured for that Pod. | +| egress.dns.config | object | `{"nameservers":[],"searches":[]}` | Optional dns configuration, required when policy is 'None' | +| egress.dns.config.nameservers | list | `[]` | A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. | +| egress.dns.config.searches | list | `[]` | A list of DNS search domains for hostname lookup in the Pod. | +| kumactl.image.repository | string | `"kumactl"` | The kumactl image repository | +| kumactl.image.tag | string | `nil` | The kumactl image tag. When not specified, the value is copied from global.tag | +| kubectl.image.registry | string | `"docker.io"` | The kubectl image registry | +| kubectl.image.repository | string | `"bitnami/kubectl"` | The kubectl image repository | +| kubectl.image.tag | string | `"1.27.5"` | The kubectl image tag | +| hooks.nodeSelector | object | `{"kubernetes.io/os":"linux"}` | Node selector for the HELM hooks | +| hooks.tolerations | list | `[]` | Tolerations for the HELM hooks | +| hooks.podSecurityContext | object | `{"runAsNonRoot":true}` | Security context at the pod level for crd/webhook/ns | +| hooks.containerSecurityContext | object | `{"readOnlyRootFilesystem":true}` | Security context at the container level for crd/webhook/ns | +| hooks.ebpfCleanup | object | `{"containerSecurityContext":{"readOnlyRootFilesystem":false},"podSecurityContext":{"runAsNonRoot":false}}` | ebpf-cleanup hook needs write access to the root filesystem to clean ebpf programs Changing below values will potentially break ebpf cleanup completely, so be cautious when doing so. | +| hooks.ebpfCleanup.podSecurityContext | object | `{"runAsNonRoot":false}` | Security context at the pod level for crd/webhook/cleanup-ebpf | +| hooks.ebpfCleanup.containerSecurityContext | object | `{"readOnlyRootFilesystem":false}` | Security context at the container level for crd/webhook/cleanup-ebpf | +| transparentProxy.configMap.enabled | bool | `false` | If true, enables the use of a ConfigMap to manage transparent proxy configuration instead of directly configuring it within the Kuma system | +| transparentProxy.configMap.name | string | `"kuma-transparent-proxy-config"` | The name of the ConfigMap used to store the transparent proxy configuration | +| transparentProxy.configMap.config.kumaDPUser | string | `"5678"` | The username or UID of the user that will run kuma-dp. If not provided, the system will use the default UID ("5678") or the default username ("kuma-dp") | +| transparentProxy.configMap.config.ipFamilyMode | string | `"dualstack"` | The IP family mode used for configuring traffic redirection in the transparent proxy Supports "dualstack" (for both IPv4 and IPv6) and "ipv4" modes | +| transparentProxy.configMap.config.redirect.dns.enabled | bool | `true` | Enables DNS redirection in the transparent proxy | +| transparentProxy.configMap.config.redirect.dns.captureAll | bool | `true` | Redirect all DNS queries | +| transparentProxy.configMap.config.redirect.dns.port | int | `15053` | The port on which the DNS server listens | +| transparentProxy.configMap.config.redirect.dns.resolvConfigPath | string | `"/etc/resolv.conf"` | Path to the system's resolv.conf file | +| transparentProxy.configMap.config.redirect.dns.skipConntrackZoneSplit | bool | `false` | Disables conntrack zone splitting, which can prevent potential DNS issues | +| transparentProxy.configMap.config.redirect.inbound.enabled | bool | `true` | Enables inbound traffic redirection | +| transparentProxy.configMap.config.redirect.inbound.port | int | `15006` | Port used for redirecting inbound traffic | +| transparentProxy.configMap.config.redirect.inbound.excludePorts | list | `[]` | List of ports to exclude from inbound traffic redirection | +| transparentProxy.configMap.config.redirect.inbound.excludePortsForIPs | list | `[]` | List of IP addresses to exclude from inbound traffic redirection for specific ports | +| transparentProxy.configMap.config.redirect.inbound.excludePortsForUIDs | list | `[]` | List of UIDs to exclude from inbound traffic redirection for specific ports | +| transparentProxy.configMap.config.redirect.inbound.includePorts | list | `[]` | List of ports to include in inbound traffic redirection | +| transparentProxy.configMap.config.redirect.inbound.insertRedirectInsteadOfAppend | bool | `false` | Inserts the redirection rule at the beginning of the chain instead of appending it | +| transparentProxy.configMap.config.redirect.outbound.enabled | bool | `true` | Enables outbound traffic redirection | +| transparentProxy.configMap.config.redirect.outbound.port | int | `15001` | Port used for redirecting outbound traffic | +| transparentProxy.configMap.config.redirect.outbound.excludePorts | list | `[]` | List of ports to exclude from outbound traffic redirection | +| transparentProxy.configMap.config.redirect.outbound.excludePortsForIPs | list | `[]` | List of IP addresses to exclude from outbound traffic redirection for specific ports | +| transparentProxy.configMap.config.redirect.outbound.excludePortsForUIDs | list | `[]` | List of UIDs to exclude from outbound traffic redirection for specific ports | +| transparentProxy.configMap.config.redirect.outbound.includePorts | list | `[]` | List of ports to include in outbound traffic redirection | +| transparentProxy.configMap.config.redirect.outbound.insertRedirectInsteadOfAppend | bool | `false` | Inserts the redirection rule at the beginning of the chain instead of appending it | +| transparentProxy.configMap.config.redirect.vnet.networks | list | `[]` | Specifies virtual networks using the format interfaceName:CIDR Allows matching traffic on specific network interfaces Examples: - "docker0:172.17.0.0/16" - "br+:172.18.0.0/16" (matches any interface starting with "br") - "iface:::1/64" (for IPv6) | +| transparentProxy.configMap.config.ebpf.enabled | bool | `false` | Enables eBPF support for handling traffic redirection in the transparent proxy | +| transparentProxy.configMap.config.ebpf.bpffsPath | string | `"/run/kuma/bpf"` | The path of the BPF filesystem | +| transparentProxy.configMap.config.ebpf.cgroupPath | string | `"/sys/fs/cgroup"` | The path of cgroup2 | +| transparentProxy.configMap.config.ebpf.instanceIPEnvVarName | string | `""` | The name of the environment variable containing the IP address of the instance (pod/vm) where transparent proxy will be installed | +| transparentProxy.configMap.config.ebpf.programsSourcePath | string | `"/tmp/kuma-ebpf"` | Path where compiled eBPF programs and other necessary files for eBPF mode can be found | +| transparentProxy.configMap.config.ebpf.tcAttachIface | string | `""` | The network interface for TC eBPF programs to bind to. If not provided, it will be automatically determined | +| transparentProxy.configMap.config.retry.maxRetries | int | `4` | The maximum number of retry attempts for operations | +| transparentProxy.configMap.config.retry.sleepBetweenRetries | string | `"2s"` | The time duration to wait between retry attempts | +| transparentProxy.configMap.config.iptablesExecutables.iptables | string | `""` | Custom path for the iptables executable (IPv4) | +| transparentProxy.configMap.config.iptablesExecutables.iptables-save | string | `""` | Custom path for the iptables-save executable (IPv4) | +| transparentProxy.configMap.config.iptablesExecutables.iptables-restore | string | `""` | Custom path for the iptables-restore executable (IPv4) | +| transparentProxy.configMap.config.iptablesExecutables.ip6tables | string | `""` | Custom path for the ip6tables executable (IPv6) | +| transparentProxy.configMap.config.iptablesExecutables.ip6tables-save | string | `""` | Custom path for the ip6tables-save executable (IPv6) | +| transparentProxy.configMap.config.iptablesExecutables.ip6tables-restore | string | `""` | Custom path for the ip6tables-restore executable (IPv6) | +| transparentProxy.configMap.config.log.enabled | bool | `false` | Enables logging of iptables rules for diagnostics and monitoring | +| transparentProxy.configMap.config.comments.disabled | bool | `false` | Disables comments in the generated iptables rules | +| transparentProxy.configMap.config.wait | int | `5` | Time in seconds to wait for acquiring the xtables lock before failing Value 0 means wait indefinitely | +| transparentProxy.configMap.config.waitInterval | int | `0` | Time interval between retries to acquire the xtables lock in seconds | +| transparentProxy.configMap.config.dropInvalidPackets | bool | `false` | Drops invalid packets to avoid connection resets in high-throughput scenarios | +| transparentProxy.configMap.config.storeFirewalld | bool | `false` | Enables firewalld support to store iptables rules | +| transparentProxy.configMap.config.verbose | bool | `false` | Enables verbose mode with longer argument/flag names and additional comments | +| experimental.ebpf.enabled | bool | `false` | If true, ebpf will be used instead of using iptables to install/configure transparent proxy | +| experimental.ebpf.instanceIPEnvVarName | string | `"INSTANCE_IP"` | Name of the environmental variable which will contain the IP address of a pod | +| experimental.ebpf.bpffsPath | string | `"/sys/fs/bpf"` | Path where BPF file system should be mounted | +| experimental.ebpf.cgroupPath | string | `"/sys/fs/cgroup"` | Host's cgroup2 path | +| experimental.ebpf.tcAttachIface | string | `""` | Name of the network interface which TC programs should be attached to, we'll try to automatically determine it if empty | +| experimental.ebpf.programsSourcePath | string | `"/tmp/kuma-ebpf"` | Path where compiled eBPF programs which will be installed can be found | +| experimental.sidecarContainers | bool | `false` | If true, enable native Kubernetes sidecars. This requires at least Kubernetes v1.29 | +| postgres.port | string | `"5432"` | Postgres port, password should be provided as a secret reference in "controlPlane.secrets" with the Env value "KUMA_STORE_POSTGRES_PASSWORD". Example: controlPlane: secrets: - Secret: postgres-postgresql Key: postgresql-password Env: KUMA_STORE_POSTGRES_PASSWORD | +| postgres.tls.mode | string | `"disable"` | Mode of TLS connection. Available values are: "disable", "verifyNone", "verifyCa", "verifyFull" | +| postgres.tls.disableSSLSNI | bool | `false` | Whether to disable SNI the postgres `sslsni` option. | +| postgres.tls.caSecretName | string | `nil` | Secret name that contains the ca.crt | +| postgres.tls.secretName | string | `nil` | Secret name that contains the client tls.crt, tls.key | + +## Custom Resource Definitions + +All Kuma CRDs are loaded via the [`crds`](crds) directory. For more detailed information on CRDs and Helm, +please refer to [the Helm documentation][helm-crd]. + +## Deleting + +As part of [Helm's limitations][helm-crd-limitations], CRDs will not be deleted when the `kuma` chart is deleted and +must be deleted manually. When a CRD is deleted Kubernetes deletes all resources of that kind as well, so this should +be done carefully. + +To do this with `kubectl` on *nix platforms, run: + +```shell +kubectl get crds | grep kuma.io | tr -s " " | cut -d " " -f1 | xargs kubectl delete crd + +# or with jq +kubectl get crds -o json | jq '.items[].metadata.name | select(.|test(".*kuma\\.io"))' | xargs kubectl delete crd +``` + +## Autoscaling + +In production, it is advisable to enable Control Plane autoscaling for High Availability. Autoscaling uses the +`HorizontalPodAutoscaler` resource to add redundancy and scale the CP pods based on CPU utilization, which requires +the [k8s metrics-server][kube-metrics-server] to be running on the cluster. + +## Development + +The charts are used internally in `kumactl install`, therefore the following rules apply when developing new chat features: + * all templates that start with `pre-` and `post-` are omitted when processing in `kumactl install` + +### Installing Metrics Server for Autoscaling + +If running on kind, or on a cluster with a similarly self-signed cert, the metrics server must be configured to allow +insecure kubelet TLS. The make task `kind/deploy/metrics-server` installs this patched version of the server. + +[kuma-url]: https://kuma.io/ +[kuma-logo]: https://kuma-public-assets.s3.amazonaws.com/kuma-logo-v2.png +[helm-crd]: https://helm.sh/docs/chart_best_practices/custom_resource_definitions/ +[helm-crd-limitations]: https://helm.sh/docs/topics/charts/#limitations-on-crds +[kube-metrics-server]: https://github.com/kubernetes-sigs/metrics-server diff --git a/charts/kuma/kuma/2.9.1/README.md.gotmpl b/charts/kuma/kuma/2.9.1/README.md.gotmpl new file mode 100644 index 000000000..3b296a411 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/README.md.gotmpl @@ -0,0 +1,52 @@ +[![][kuma-logo]][kuma-url] + +{{ template "chart.description" . }} + +{{ template "chart.typeBadge" . }}{{ template "chart.versionBadge" . }}{{ template "chart.appVersionBadge" . }} + +{{ template "chart.homepageLine" . }} + +{{ template "chart.valuesSection" . }} + +## Custom Resource Definitions + +All Kuma CRDs are loaded via the [`crds`](crds) directory. For more detailed information on CRDs and Helm, +please refer to [the Helm documentation][helm-crd]. + +## Deleting + +As part of [Helm's limitations][helm-crd-limitations], CRDs will not be deleted when the `kuma` chart is deleted and +must be deleted manually. When a CRD is deleted Kubernetes deletes all resources of that kind as well, so this should +be done carefully. + +To do this with `kubectl` on *nix platforms, run: + +```shell +kubectl get crds | grep kuma.io | tr -s " " | cut -d " " -f1 | xargs kubectl delete crd + +# or with jq +kubectl get crds -o json | jq '.items[].metadata.name | select(.|test(".*kuma\\.io"))' | xargs kubectl delete crd +``` + +## Autoscaling + +In production, it is advisable to enable Control Plane autoscaling for High Availability. Autoscaling uses the +`HorizontalPodAutoscaler` resource to add redundancy and scale the CP pods based on CPU utilization, which requires +the [k8s metrics-server][kube-metrics-server] to be running on the cluster. + +## Development + +The charts are used internally in `kumactl install`, therefore the following rules apply when developing new chat features: + * all templates that start with `pre-` and `post-` are omitted when processing in `kumactl install` + +### Installing Metrics Server for Autoscaling + +If running on kind, or on a cluster with a similarly self-signed cert, the metrics server must be configured to allow +insecure kubelet TLS. The make task `kind/deploy/metrics-server` installs this patched version of the server. + + +[kuma-url]: https://kuma.io/ +[kuma-logo]: https://kuma-public-assets.s3.amazonaws.com/kuma-logo-v2.png +[helm-crd]: https://helm.sh/docs/chart_best_practices/custom_resource_definitions/ +[helm-crd-limitations]: https://helm.sh/docs/topics/charts/#limitations-on-crds +[kube-metrics-server]: https://github.com/kubernetes-sigs/metrics-server diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_circuitbreakers.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_circuitbreakers.yaml new file mode 100644 index 000000000..ea955f2ab --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_circuitbreakers.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: circuitbreakers.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: CircuitBreaker + listKind: CircuitBreakerList + plural: circuitbreakers + singular: circuitbreaker + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma CircuitBreaker resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_containerpatches.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_containerpatches.yaml new file mode 100644 index 000000000..9fc77a966 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_containerpatches.yaml @@ -0,0 +1,114 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: containerpatches.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ContainerPatch + listKind: ContainerPatchList + plural: containerpatches + singular: containerpatch + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: ContainerPatch stores a list of patches to apply to init and + sidecar containers. + 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 + mesh: + type: string + metadata: + type: object + spec: + description: ContainerPatchSpec specifies the options available for a + ContainerPatch + properties: + initPatch: + description: InitPatch specifies jsonpatch to apply to an init container. + items: + description: JsonPatchBlock is one json patch operation block. + properties: + from: + description: From is a jsonpatch from string, used by move and + copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: |- + Value must be a string representing a valid json object used + by replace and add operations. String has to be escaped with " to be valid a json object. + type: string + required: + - op + - path + type: object + type: array + sidecarPatch: + description: SidecarPatch specifies jsonpatch to apply to a sidecar + container. + items: + description: JsonPatchBlock is one json patch operation block. + properties: + from: + description: From is a jsonpatch from string, used by move and + copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: |- + Value must be a string representing a valid json object used + by replace and add operations. String has to be escaped with " to be valid a json object. + type: string + required: + - op + - path + type: object + type: array + type: object + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplaneinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplaneinsights.yaml new file mode 100644 index 000000000..23c4538ea --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplaneinsights.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: dataplaneinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: DataplaneInsight + listKind: DataplaneInsightList + plural: dataplaneinsights + singular: dataplaneinsight + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + status: + description: Status is the status the Kuma resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplanes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplanes.yaml new file mode 100644 index 000000000..ec8f06342 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_dataplanes.yaml @@ -0,0 +1,70 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: dataplanes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: Dataplane + listKind: DataplaneList + plural: dataplanes + singular: dataplane + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Service tag of the first inbound + jsonPath: .spec.networking.inbound[0].tags['kuma\.io/service'] + name: kuma.io/service + type: string + - description: Service tag of the second inbound + jsonPath: .spec.networking.inbound[1].tags['kuma\.io/service'] + name: kuma.io/service + type: string + - description: Service tag of the third inbound + jsonPath: .spec.networking.inbound[2].tags['kuma\.io/service'] + name: kuma.io/service + priority: 1 + type: string + - description: Service tag of the fourth inbound + jsonPath: .spec.networking.inbound[3].tags['kuma\.io/service'] + name: kuma.io/service + priority: 1 + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma Dataplane resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_externalservices.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_externalservices.yaml new file mode 100644 index 000000000..be37a7b7f --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_externalservices.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: externalservices.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ExternalService + listKind: ExternalServiceList + plural: externalservices + singular: externalservice + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ExternalService resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_faultinjections.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_faultinjections.yaml new file mode 100644 index 000000000..6fb6366d5 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_faultinjections.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: faultinjections.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: FaultInjection + listKind: FaultInjectionList + plural: faultinjections + singular: faultinjection + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma FaultInjection resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_healthchecks.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_healthchecks.yaml new file mode 100644 index 000000000..9f2d075b5 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_healthchecks.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: healthchecks.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: HealthCheck + listKind: HealthCheckList + plural: healthchecks + singular: healthcheck + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma HealthCheck resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_hostnamegenerators.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_hostnamegenerators.yaml new file mode 100644 index 000000000..943421775 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_hostnamegenerators.yaml @@ -0,0 +1,72 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: hostnamegenerators.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: HostnameGenerator + listKind: HostnameGeneratorList + plural: hostnamegenerators + singular: hostnamegenerator + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma HostnameGenerator resource. + properties: + selector: + properties: + meshExternalService: + properties: + matchLabels: + additionalProperties: + type: string + type: object + type: object + meshMultiZoneService: + properties: + matchLabels: + additionalProperties: + type: string + type: object + type: object + meshService: + properties: + matchLabels: + additionalProperties: + type: string + type: object + type: object + type: object + template: + type: string + type: object + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshaccesslogs.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshaccesslogs.yaml new file mode 100644 index 000000000..16191c5ba --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshaccesslogs.yaml @@ -0,0 +1,557 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshaccesslogs.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshAccessLog + listKind: MeshAccessLogList + plural: meshaccesslogs + singular: meshaccesslog + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshAccessLog resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + backends: + items: + properties: + file: + description: FileBackend defines configuration for + file based access logs + properties: + format: + description: |- + Format of access logs. Placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + properties: + json: + example: + - key: start_time + value: '%START_TIME%' + - key: bytes_received + value: '%BYTES_RECEIVED%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + omitEmptyValues: + default: false + type: boolean + plain: + example: '[%START_TIME%] %KUMA_MESH% %UPSTREAM_HOST%' + type: string + type: + enum: + - Plain + - Json + type: string + required: + - type + type: object + path: + description: Path to a file that logs will be + written to + example: /tmp/access.log + minLength: 1 + type: string + required: + - path + type: object + openTelemetry: + description: Defines an OpenTelemetry logging backend. + properties: + attributes: + description: |- + Attributes can contain placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + example: + - key: mesh + value: '%KUMA_MESH%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + body: + description: |- + Body is a raw string or an OTLP any value as described at + https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/logs/data-model.md#field-body + It can contain placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + example: + kvlistValue: + values: + - key: mesh + value: + stringValue: '%KUMA_MESH%' + x-kubernetes-preserve-unknown-fields: true + endpoint: + description: Endpoint of OpenTelemetry collector. + An empty port defaults to 4317. + example: otel-collector:4317 + minLength: 1 + type: string + required: + - endpoint + type: object + tcp: + description: TCPBackend defines a TCP logging backend. + properties: + address: + description: Address of the TCP logging backend + example: 127.0.0.1:5000 + minLength: 1 + type: string + format: + description: |- + Format of access logs. Placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + properties: + json: + example: + - key: start_time + value: '%START_TIME%' + - key: bytes_received + value: '%BYTES_RECEIVED%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + omitEmptyValues: + default: false + type: boolean + plain: + example: '[%START_TIME%] %KUMA_MESH% %UPSTREAM_HOST%' + type: string + type: + enum: + - Plain + - Json + type: string + required: + - type + type: object + required: + - address + type: object + type: + enum: + - Tcp + - File + - OpenTelemetry + type: string + required: + - type + type: object + type: array + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in-place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between the consumed services and + corresponding configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + backends: + items: + properties: + file: + description: FileBackend defines configuration for + file based access logs + properties: + format: + description: |- + Format of access logs. Placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + properties: + json: + example: + - key: start_time + value: '%START_TIME%' + - key: bytes_received + value: '%BYTES_RECEIVED%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + omitEmptyValues: + default: false + type: boolean + plain: + example: '[%START_TIME%] %KUMA_MESH% %UPSTREAM_HOST%' + type: string + type: + enum: + - Plain + - Json + type: string + required: + - type + type: object + path: + description: Path to a file that logs will be + written to + example: /tmp/access.log + minLength: 1 + type: string + required: + - path + type: object + openTelemetry: + description: Defines an OpenTelemetry logging backend. + properties: + attributes: + description: |- + Attributes can contain placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + example: + - key: mesh + value: '%KUMA_MESH%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + body: + description: |- + Body is a raw string or an OTLP any value as described at + https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/logs/data-model.md#field-body + It can contain placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + example: + kvlistValue: + values: + - key: mesh + value: + stringValue: '%KUMA_MESH%' + x-kubernetes-preserve-unknown-fields: true + endpoint: + description: Endpoint of OpenTelemetry collector. + An empty port defaults to 4317. + example: otel-collector:4317 + minLength: 1 + type: string + required: + - endpoint + type: object + tcp: + description: TCPBackend defines a TCP logging backend. + properties: + address: + description: Address of the TCP logging backend + example: 127.0.0.1:5000 + minLength: 1 + type: string + format: + description: |- + Format of access logs. Placeholders available on + https://www.envoyproxy.io/docs/envoy/latest/configuration/observability/access_log/usage#command-operators + properties: + json: + example: + - key: start_time + value: '%START_TIME%' + - key: bytes_received + value: '%BYTES_RECEIVED%' + items: + properties: + key: + type: string + value: + type: string + type: object + type: array + omitEmptyValues: + default: false + type: boolean + plain: + example: '[%START_TIME%] %KUMA_MESH% %UPSTREAM_HOST%' + type: string + type: + enum: + - Plain + - Json + type: string + required: + - type + type: object + required: + - address + type: object + type: + enum: + - Tcp + - File + - OpenTelemetry + type: string + required: + - type + type: object + type: array + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshcircuitbreakers.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshcircuitbreakers.yaml new file mode 100644 index 000000000..bea1fb597 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshcircuitbreakers.yaml @@ -0,0 +1,739 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshcircuitbreakers.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshCircuitBreaker + listKind: MeshCircuitBreakerList + plural: meshcircuitbreakers + singular: meshcircuitbreaker + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshCircuitBreaker + resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations + referenced in 'targetRef' + properties: + connectionLimits: + description: |- + ConnectionLimits contains configuration of each circuit breaking limit, + which when exceeded makes the circuit breaker to become open (no traffic + is allowed like no current is allowed in the circuits when physical + circuit breaker ir open) + properties: + maxConnectionPools: + description: |- + The maximum number of connection pools per cluster that are concurrently + supported at once. Set this for clusters which create a large number of + connection pools. + format: int32 + type: integer + maxConnections: + description: |- + The maximum number of connections allowed to be made to the upstream + cluster. + format: int32 + type: integer + maxPendingRequests: + description: |- + The maximum number of pending requests that are allowed to the upstream + cluster. This limit is applied as a connection limit for non-HTTP + traffic. + format: int32 + type: integer + maxRequests: + description: |- + The maximum number of parallel requests that are allowed to be made + to the upstream cluster. This limit does not apply to non-HTTP traffic. + format: int32 + type: integer + maxRetries: + description: |- + The maximum number of parallel retries that will be allowed to + the upstream cluster. + format: int32 + type: integer + type: object + outlierDetection: + description: |- + OutlierDetection contains the configuration of the process of dynamically + determining whether some number of hosts in an upstream cluster are + performing unlike the others and removing them from the healthy load + balancing set. Performance might be along different axes such as + consecutive failures, temporal success rate, temporal latency, etc. + Outlier detection is a form of passive health checking. + properties: + baseEjectionTime: + description: |- + The base time that a host is ejected for. The real time is equal to + the base time multiplied by the number of times the host has been + ejected. + type: string + detectors: + description: Contains configuration for supported outlier + detectors + properties: + failurePercentage: + description: |- + Failure Percentage based outlier detection functions similarly to success + rate detection, in that it relies on success rate data from each host in + a cluster. However, rather than compare those values to the mean success + rate of the cluster as a whole, they are compared to a flat + user-configured threshold. This threshold is configured via the + outlierDetection.failurePercentageThreshold field. + The other configuration fields for failure percentage based detection are + similar to the fields for success rate detection. As with success rate + detection, detection will not be performed for a host if its request + volume over the aggregation interval is less than the + outlierDetection.detectors.failurePercentage.requestVolume value. + Detection also will not be performed for a cluster if the number of hosts + with the minimum required request volume in an interval is less than the + outlierDetection.detectors.failurePercentage.minimumHosts value. + properties: + minimumHosts: + description: |- + The minimum number of hosts in a cluster in order to perform failure + percentage-based ejection. If the total number of hosts in the cluster is + less than this value, failure percentage-based ejection will not be + performed. + format: int32 + type: integer + requestVolume: + description: |- + The minimum number of total requests that must be collected in one + interval (as defined by the interval duration above) to perform failure + percentage-based ejection for this host. If the volume is lower than this + setting, failure percentage-based ejection will not be performed for this + host. + format: int32 + type: integer + threshold: + description: |- + The failure percentage to use when determining failure percentage-based + outlier detection. If the failure percentage of a given host is greater + than or equal to this value, it will be ejected. + format: int32 + type: integer + type: object + gatewayFailures: + description: |- + In the default mode (outlierDetection.splitExternalLocalOriginErrors is + false) this detection type takes into account a subset of 5xx errors, + called "gateway errors" (502, 503 or 504 status code) and local origin + failures, such as timeout, TCP reset etc. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true) + this detection type takes into account a subset of 5xx errors, called + "gateway errors" (502, 503 or 504 status code) and is supported only by + the http router. + properties: + consecutive: + description: |- + The number of consecutive gateway failures (502, 503, 504 status codes) + before a consecutive gateway failure ejection occurs. + format: int32 + type: integer + type: object + localOriginFailures: + description: |- + This detection type is enabled only when + outlierDetection.splitExternalLocalOriginErrors is true and takes into + account only locally originated errors (timeout, reset, etc). + If Envoy repeatedly cannot connect to an upstream host or communication + with the upstream host is repeatedly interrupted, it will be ejected. + Various locally originated problems are detected: timeout, TCP reset, + ICMP errors, etc. This detection type is supported by http router and + tcp proxy. + properties: + consecutive: + description: |- + The number of consecutive locally originated failures before ejection + occurs. Parameter takes effect only when splitExternalAndLocalErrors + is set to true. + format: int32 + type: integer + type: object + successRate: + description: |- + Success Rate based outlier detection aggregates success rate data from + every host in a cluster. Then at given intervals ejects hosts based on + statistical outlier detection. Success Rate outlier detection will not be + calculated for a host if its request volume over the aggregation interval + is less than the outlierDetection.detectors.successRate.requestVolume + value. + Moreover, detection will not be performed for a cluster if the number of + hosts with the minimum required request volume in an interval is less + than the outlierDetection.detectors.successRate.minimumHosts value. + In the default configuration mode + (outlierDetection.splitExternalLocalOriginErrors is false) this detection + type takes into account all types of errors: locally and externally + originated. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true), + locally originated errors and externally originated (transaction) errors + are counted and treated separately. + properties: + minimumHosts: + description: |- + The number of hosts in a cluster that must have enough request volume to + detect success rate outliers. If the number of hosts is less than this + setting, outlier detection via success rate statistics is not performed + for any host in the cluster. + format: int32 + type: integer + requestVolume: + description: |- + The minimum number of total requests that must be collected in one + interval (as defined by the interval duration configured in + outlierDetection section) to include this host in success rate based + outlier detection. If the volume is lower than this setting, outlier + detection via success rate statistics is not performed for that host. + format: int32 + type: integer + standardDeviationFactor: + anyOf: + - type: integer + - type: string + description: |- + This factor is used to determine the ejection threshold for success rate + outlier ejection. The ejection threshold is the difference between + the mean success rate, and the product of this factor and the standard + deviation of the mean success rate: mean - (standard_deviation * + success_rate_standard_deviation_factor). + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + type: object + totalFailures: + description: |- + In the default mode (outlierDetection.splitExternalAndLocalErrors is + false) this detection type takes into account all generated errors: + locally originated and externally originated (transaction) errors. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true) + this detection type takes into account only externally originated + (transaction) errors, ignoring locally originated errors. + If an upstream host is an HTTP-server, only 5xx types of error are taken + into account (see Consecutive Gateway Failure for exceptions). + Properly formatted responses, even when they carry an operational error + (like index not found, access denied) are not taken into account. + properties: + consecutive: + description: |- + The number of consecutive server-side error responses (for HTTP traffic, + 5xx responses; for TCP traffic, connection failures; for Redis, failure + to respond PONG; etc.) before a consecutive total failure ejection + occurs. + format: int32 + type: integer + type: object + type: object + disabled: + description: When set to true, outlierDetection configuration + won't take any effect + type: boolean + interval: + description: |- + The time interval between ejection analysis sweeps. This can result in + both new ejections and hosts being returned to service. + type: string + maxEjectionPercent: + description: |- + The maximum % of an upstream cluster that can be ejected due to outlier + detection. Defaults to 10% but will eject at least one host regardless of + the value. + format: int32 + type: integer + splitExternalAndLocalErrors: + description: |- + Determines whether to distinguish local origin failures from external + errors. If set to true the following configuration parameters are taken + into account: detectors.localOriginFailures.consecutive + type: boolean + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: |- + To list makes a match between the consumed services and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations + referenced in 'targetRef' + properties: + connectionLimits: + description: |- + ConnectionLimits contains configuration of each circuit breaking limit, + which when exceeded makes the circuit breaker to become open (no traffic + is allowed like no current is allowed in the circuits when physical + circuit breaker ir open) + properties: + maxConnectionPools: + description: |- + The maximum number of connection pools per cluster that are concurrently + supported at once. Set this for clusters which create a large number of + connection pools. + format: int32 + type: integer + maxConnections: + description: |- + The maximum number of connections allowed to be made to the upstream + cluster. + format: int32 + type: integer + maxPendingRequests: + description: |- + The maximum number of pending requests that are allowed to the upstream + cluster. This limit is applied as a connection limit for non-HTTP + traffic. + format: int32 + type: integer + maxRequests: + description: |- + The maximum number of parallel requests that are allowed to be made + to the upstream cluster. This limit does not apply to non-HTTP traffic. + format: int32 + type: integer + maxRetries: + description: |- + The maximum number of parallel retries that will be allowed to + the upstream cluster. + format: int32 + type: integer + type: object + outlierDetection: + description: |- + OutlierDetection contains the configuration of the process of dynamically + determining whether some number of hosts in an upstream cluster are + performing unlike the others and removing them from the healthy load + balancing set. Performance might be along different axes such as + consecutive failures, temporal success rate, temporal latency, etc. + Outlier detection is a form of passive health checking. + properties: + baseEjectionTime: + description: |- + The base time that a host is ejected for. The real time is equal to + the base time multiplied by the number of times the host has been + ejected. + type: string + detectors: + description: Contains configuration for supported outlier + detectors + properties: + failurePercentage: + description: |- + Failure Percentage based outlier detection functions similarly to success + rate detection, in that it relies on success rate data from each host in + a cluster. However, rather than compare those values to the mean success + rate of the cluster as a whole, they are compared to a flat + user-configured threshold. This threshold is configured via the + outlierDetection.failurePercentageThreshold field. + The other configuration fields for failure percentage based detection are + similar to the fields for success rate detection. As with success rate + detection, detection will not be performed for a host if its request + volume over the aggregation interval is less than the + outlierDetection.detectors.failurePercentage.requestVolume value. + Detection also will not be performed for a cluster if the number of hosts + with the minimum required request volume in an interval is less than the + outlierDetection.detectors.failurePercentage.minimumHosts value. + properties: + minimumHosts: + description: |- + The minimum number of hosts in a cluster in order to perform failure + percentage-based ejection. If the total number of hosts in the cluster is + less than this value, failure percentage-based ejection will not be + performed. + format: int32 + type: integer + requestVolume: + description: |- + The minimum number of total requests that must be collected in one + interval (as defined by the interval duration above) to perform failure + percentage-based ejection for this host. If the volume is lower than this + setting, failure percentage-based ejection will not be performed for this + host. + format: int32 + type: integer + threshold: + description: |- + The failure percentage to use when determining failure percentage-based + outlier detection. If the failure percentage of a given host is greater + than or equal to this value, it will be ejected. + format: int32 + type: integer + type: object + gatewayFailures: + description: |- + In the default mode (outlierDetection.splitExternalLocalOriginErrors is + false) this detection type takes into account a subset of 5xx errors, + called "gateway errors" (502, 503 or 504 status code) and local origin + failures, such as timeout, TCP reset etc. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true) + this detection type takes into account a subset of 5xx errors, called + "gateway errors" (502, 503 or 504 status code) and is supported only by + the http router. + properties: + consecutive: + description: |- + The number of consecutive gateway failures (502, 503, 504 status codes) + before a consecutive gateway failure ejection occurs. + format: int32 + type: integer + type: object + localOriginFailures: + description: |- + This detection type is enabled only when + outlierDetection.splitExternalLocalOriginErrors is true and takes into + account only locally originated errors (timeout, reset, etc). + If Envoy repeatedly cannot connect to an upstream host or communication + with the upstream host is repeatedly interrupted, it will be ejected. + Various locally originated problems are detected: timeout, TCP reset, + ICMP errors, etc. This detection type is supported by http router and + tcp proxy. + properties: + consecutive: + description: |- + The number of consecutive locally originated failures before ejection + occurs. Parameter takes effect only when splitExternalAndLocalErrors + is set to true. + format: int32 + type: integer + type: object + successRate: + description: |- + Success Rate based outlier detection aggregates success rate data from + every host in a cluster. Then at given intervals ejects hosts based on + statistical outlier detection. Success Rate outlier detection will not be + calculated for a host if its request volume over the aggregation interval + is less than the outlierDetection.detectors.successRate.requestVolume + value. + Moreover, detection will not be performed for a cluster if the number of + hosts with the minimum required request volume in an interval is less + than the outlierDetection.detectors.successRate.minimumHosts value. + In the default configuration mode + (outlierDetection.splitExternalLocalOriginErrors is false) this detection + type takes into account all types of errors: locally and externally + originated. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true), + locally originated errors and externally originated (transaction) errors + are counted and treated separately. + properties: + minimumHosts: + description: |- + The number of hosts in a cluster that must have enough request volume to + detect success rate outliers. If the number of hosts is less than this + setting, outlier detection via success rate statistics is not performed + for any host in the cluster. + format: int32 + type: integer + requestVolume: + description: |- + The minimum number of total requests that must be collected in one + interval (as defined by the interval duration configured in + outlierDetection section) to include this host in success rate based + outlier detection. If the volume is lower than this setting, outlier + detection via success rate statistics is not performed for that host. + format: int32 + type: integer + standardDeviationFactor: + anyOf: + - type: integer + - type: string + description: |- + This factor is used to determine the ejection threshold for success rate + outlier ejection. The ejection threshold is the difference between + the mean success rate, and the product of this factor and the standard + deviation of the mean success rate: mean - (standard_deviation * + success_rate_standard_deviation_factor). + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + type: object + totalFailures: + description: |- + In the default mode (outlierDetection.splitExternalAndLocalErrors is + false) this detection type takes into account all generated errors: + locally originated and externally originated (transaction) errors. + In split mode (outlierDetection.splitExternalLocalOriginErrors is true) + this detection type takes into account only externally originated + (transaction) errors, ignoring locally originated errors. + If an upstream host is an HTTP-server, only 5xx types of error are taken + into account (see Consecutive Gateway Failure for exceptions). + Properly formatted responses, even when they carry an operational error + (like index not found, access denied) are not taken into account. + properties: + consecutive: + description: |- + The number of consecutive server-side error responses (for HTTP traffic, + 5xx responses; for TCP traffic, connection failures; for Redis, failure + to respond PONG; etc.) before a consecutive total failure ejection + occurs. + format: int32 + type: integer + type: object + type: object + disabled: + description: When set to true, outlierDetection configuration + won't take any effect + type: boolean + interval: + description: |- + The time interval between ejection analysis sweeps. This can result in + both new ejections and hosts being returned to service. + type: string + maxEjectionPercent: + description: |- + The maximum % of an upstream cluster that can be ejected due to outlier + detection. Defaults to 10% but will eject at least one host regardless of + the value. + format: int32 + type: integer + splitExternalAndLocalErrors: + description: |- + Determines whether to distinguish local origin failures from external + errors. If set to true the following configuration parameters are taken + into account: detectors.localOriginFailures.consecutive + type: boolean + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshes.yaml new file mode 100644 index 000000000..a9fec649c --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshes.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: Mesh + listKind: MeshList + plural: meshes + singular: mesh + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma Mesh resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshexternalservices.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshexternalservices.yaml new file mode 100644 index 000000000..12f87ab5a --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshexternalservices.yaml @@ -0,0 +1,333 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshexternalservices.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshExternalService + listKind: MeshExternalServiceList + plural: meshexternalservices + singular: meshexternalservice + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.addresses[0].hostname + name: Hostname + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshExternalService + resource. + properties: + endpoints: + description: Endpoints defines a list of destinations to send traffic + to. + items: + properties: + address: + description: Address defines an address to which a user want + to send a request. Is possible to provide `domain`, `ip`. + example: example.com + minLength: 1 + type: string + port: + description: Port of the endpoint + maximum: 65535 + minimum: 1 + type: integer + required: + - address + - port + type: object + type: array + extension: + description: Extension struct for a plugin configuration, in the presence + of an extension `endpoints` and `tls` are not required anymore - + it's up to the extension to validate them independently. + properties: + config: + description: Config freeform configuration for the extension. + x-kubernetes-preserve-unknown-fields: true + type: + description: Type of the extension. + type: string + required: + - config + - type + type: object + match: + description: Match defines traffic that should be routed through the + sidecar. + properties: + port: + description: Port defines a port to which a user does request. + maximum: 65535 + minimum: 1 + type: integer + protocol: + default: tcp + description: 'Protocol defines a protocol of the communication. + Possible values: `tcp`, `grpc`, `http`, `http2`.' + enum: + - tcp + - grpc + - http + - http2 + type: string + type: + default: HostnameGenerator + description: Type of the match, only `HostnameGenerator` is available + at the moment. + enum: + - HostnameGenerator + type: string + required: + - port + type: object + tls: + description: Tls provides a TLS configuration when proxy is resposible + for a TLS origination + properties: + allowRenegotiation: + default: false + description: |- + AllowRenegotiation defines if TLS sessions will allow renegotiation. + Setting this to true is not recommended for security reasons. + type: boolean + enabled: + default: false + description: Enabled defines if proxy should originate TLS. + type: boolean + verification: + description: Verification section for providing TLS verification + details. + properties: + caCert: + description: CaCert defines a certificate of CA. + properties: + inline: + description: Data source is inline bytes. + format: byte + type: string + inlineString: + description: Data source is inline string` + type: string + secret: + description: Data source is a secret with given Secret + key. + type: string + type: object + clientCert: + description: ClientCert defines a certificate of a client. + properties: + inline: + description: Data source is inline bytes. + format: byte + type: string + inlineString: + description: Data source is inline string` + type: string + secret: + description: Data source is a secret with given Secret + key. + type: string + type: object + clientKey: + description: ClientKey defines a client private key. + properties: + inline: + description: Data source is inline bytes. + format: byte + type: string + inlineString: + description: Data source is inline string` + type: string + secret: + description: Data source is a secret with given Secret + key. + type: string + type: object + mode: + default: Secured + description: Mode defines if proxy should skip verification, + one of `SkipSAN`, `SkipCA`, `Secured`, `SkipAll`. Default + `Secured`. + enum: + - SkipSAN + - SkipCA + - Secured + - SkipAll + type: string + serverName: + description: ServerName overrides the default Server Name + Indicator set by Kuma. + type: string + subjectAltNames: + description: SubjectAltNames list of names to verify in the + certificate. + items: + properties: + type: + default: Exact + description: 'Type specifies matching type, one of `Exact`, + `Prefix`. Default: `Exact`' + enum: + - Exact + - Prefix + type: string + value: + description: Value to match. + type: string + required: + - value + type: object + type: array + type: object + version: + description: Version section for providing version specification. + properties: + max: + default: TLSAuto + description: Max defines maximum supported version. One of + `TLSAuto`, `TLS10`, `TLS11`, `TLS12`, `TLS13`. + enum: + - TLSAuto + - TLS10 + - TLS11 + - TLS12 + - TLS13 + type: string + min: + default: TLSAuto + description: Min defines minimum supported version. One of + `TLSAuto`, `TLS10`, `TLS11`, `TLS12`, `TLS13`. + enum: + - TLSAuto + - TLS10 + - TLS11 + - TLS12 + - TLS13 + type: string + type: object + type: object + required: + - match + type: object + status: + description: Status is the current status of the Kuma MeshExternalService + resource. + properties: + addresses: + description: Addresses section for generated domains + items: + properties: + hostname: + type: string + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + origin: + type: string + type: object + type: array + hostnameGenerators: + items: + properties: + conditions: + description: Conditions is an array of hostname generator conditions. + items: + properties: + message: + description: |- + message is a human readable message indicating details about the transition. + This may be an empty string. + maxLength: 32768 + 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 + 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. + 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: + - message + - reason + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + required: + - hostnameGeneratorRef + type: object + type: array + vip: + description: Vip section for allocated IP + properties: + ip: + description: Value allocated IP for a provided domain with `HostnameGenerator` + type in a match section. + type: string + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshfaultinjections.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshfaultinjections.yaml new file mode 100644 index 000000000..538675b6e --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshfaultinjections.yaml @@ -0,0 +1,420 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshfaultinjections.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshFaultInjection + listKind: MeshFaultInjectionList + plural: meshfaultinjections + singular: meshfaultinjection + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshFaultInjection + resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + http: + description: Http allows to define list of Http faults between + dataplanes. + items: + description: FaultInjection defines the configuration + of faults between dataplanes. + properties: + abort: + description: |- + Abort defines a configuration of not delivering requests to destination + service and replacing the responses from destination dataplane by + predefined status code + properties: + httpStatus: + description: HTTP status code which will be returned + to source side + format: int32 + type: integer + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which abort will be injected, has to be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + required: + - httpStatus + - percentage + type: object + delay: + description: Delay defines configuration of delaying + a response from a destination + properties: + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which delay will be injected, has to be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + value: + description: The duration during which the response + will be delayed + type: string + required: + - percentage + - value + type: object + responseBandwidth: + description: |- + ResponseBandwidth defines a configuration to limit the speed of + responding to the requests + properties: + limit: + description: |- + Limit is represented by value measure in Gbps, Mbps, kbps, e.g. + 10kbps + type: string + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which response bandwidth limit will be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + required: + - limit + - percentage + type: object + type: object + type: array + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + http: + description: Http allows to define list of Http faults between + dataplanes. + items: + description: FaultInjection defines the configuration + of faults between dataplanes. + properties: + abort: + description: |- + Abort defines a configuration of not delivering requests to destination + service and replacing the responses from destination dataplane by + predefined status code + properties: + httpStatus: + description: HTTP status code which will be returned + to source side + format: int32 + type: integer + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which abort will be injected, has to be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + required: + - httpStatus + - percentage + type: object + delay: + description: Delay defines configuration of delaying + a response from a destination + properties: + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which delay will be injected, has to be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + value: + description: The duration during which the response + will be delayed + type: string + required: + - percentage + - value + type: object + responseBandwidth: + description: |- + ResponseBandwidth defines a configuration to limit the speed of + responding to the requests + properties: + limit: + description: |- + Limit is represented by value measure in Gbps, Mbps, kbps, e.g. + 10kbps + type: string + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests on which response bandwidth limit will be + either int or decimal represented as string. + x-kubernetes-int-or-string: true + required: + - limit + - percentage + type: object + type: object + type: array + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayinstances.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayinstances.yaml new file mode 100644 index 000000000..f68545cf0 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayinstances.yaml @@ -0,0 +1,354 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshgatewayinstances.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshGatewayInstance + listKind: MeshGatewayInstanceList + plural: meshgatewayinstances + singular: meshgatewayinstance + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + MeshGatewayInstance represents a managed instance of a dataplane proxy for a Kuma + Gateway. + 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: MeshGatewayInstanceSpec specifies the options available for + a GatewayDataplane. + properties: + podTemplate: + description: PodTemplate configures the Pod owned by this config. + properties: + metadata: + description: Metadata holds metadata configuration for a Service. + properties: + annotations: + additionalProperties: + type: string + description: Annotations holds annotations to be set on an + object. + type: object + labels: + additionalProperties: + type: string + description: Labels holds labels to be set on an objects. + type: object + type: object + spec: + description: Spec holds some customizable fields of a Pod. + properties: + container: + description: Container corresponds to PodSpec.Container + properties: + securityContext: + description: ContainerSecurityContext corresponds to PodSpec.Container.SecurityContext + properties: + readOnlyRootFilesystem: + description: ReadOnlyRootFilesystem corresponds to + PodSpec.Container.SecurityContext.ReadOnlyRootFilesystem + type: boolean + type: object + type: object + securityContext: + description: PodSecurityContext corresponds to PodSpec.SecurityContext + properties: + fsGroup: + description: FSGroup corresponds to PodSpec.SecurityContext.FSGroup + format: int64 + type: integer + type: object + serviceAccountName: + description: ServiceAccountName corresponds to PodSpec.ServiceAccountName. + type: string + type: object + type: object + replicas: + default: 1 + description: |- + Replicas is the number of dataplane proxy replicas to create. For + now this is a fixed number, but in the future it could be + automatically scaled based on metrics. + format: int32 + minimum: 1 + type: integer + resources: + description: |- + Resources specifies the compute resources for the proxy container. + The default can be set in the control plane config. + properties: + 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 + request: + description: |- + Request is the name chosen for a request in the referenced claim. + If empty, everything from the claim is made available, otherwise + only the result of this 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 + description: |- + Limits describes the maximum amount of compute resources allowed. + More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/ + 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 + 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 + type: object + serviceTemplate: + description: ServiceTemplate configures the Service owned by this + config. + properties: + metadata: + description: Metadata holds metadata configuration for a Service. + properties: + annotations: + additionalProperties: + type: string + description: Annotations holds annotations to be set on an + object. + type: object + labels: + additionalProperties: + type: string + description: Labels holds labels to be set on an objects. + type: object + type: object + spec: + description: Spec holds some customizable fields of a Service. + properties: + loadBalancerIP: + description: LoadBalancerIP corresponds to ServiceSpec.LoadBalancerIP. + type: string + type: object + type: object + serviceType: + default: LoadBalancer + description: |- + ServiceType specifies the type of managed Service that will be + created to expose the dataplane proxies to traffic from outside + the cluster. The ports to expose will be taken from the matching Gateway + resource. If there is no matching Gateway, the managed Service will + be deleted. + enum: + - LoadBalancer + - ClusterIP + - NodePort + type: string + tags: + additionalProperties: + type: string + description: |- + Tags specifies the Kuma tags that are propagated to the managed + dataplane proxies. These tags should not include `kuma.io/service` tag + since is auto-generated, and should match exactly one Gateway + resource. + type: object + type: object + status: + description: |- + MeshGatewayInstanceStatus holds information about the status of the gateway + instance. + properties: + conditions: + description: Conditions is an array of gateway instance conditions. + items: + description: Condition contains details for one aspect of the current + state of this API Resource. + 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. + 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 + loadBalancer: + description: |- + LoadBalancer contains the current status of the load-balancer, + if one is present. + properties: + ingress: + description: |- + Ingress is a list containing ingress points for the load-balancer. + Traffic intended for the service should be sent to these ingress points. + items: + description: |- + LoadBalancerIngress represents the status of a load-balancer ingress point: + traffic intended for the service should be sent to an ingress point. + properties: + hostname: + description: |- + Hostname is set for load-balancer ingress points that are DNS based + (typically AWS load-balancers) + type: string + ip: + description: |- + IP is set for load-balancer ingress points that are IP based + (typically GCE or OpenStack load-balancers) + type: string + ipMode: + description: |- + IPMode specifies how the load-balancer IP behaves, and may only be specified when the ip field is specified. + Setting this to "VIP" indicates that traffic is delivered to the node with + the destination set to the load-balancer's IP and port. + Setting this to "Proxy" indicates that traffic is delivered to the node or pod with + the destination set to the node's IP and node port or the pod's IP and port. + Service implementations may use this information to adjust traffic routing. + type: string + ports: + description: |- + Ports is a list of records of service ports + If used, every port defined in the service should have an entry in it + items: + properties: + error: + description: |- + Error is to record the problem with the service port + The format of the error shall comply with the following rules: + - built-in error values shall be specified in this file and those shall use + CamelCase names + - cloud provider specific error values must have names that comply with the + format 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 + port: + description: Port is the port number of the service + port of which status is recorded here + format: int32 + type: integer + protocol: + description: |- + Protocol is the protocol of the service port of which status is recorded here + The supported values are: "TCP", "UDP", "SCTP" + type: string + required: + - error + - port + - protocol + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: array + x-kubernetes-list-type: atomic + type: object + type: object + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayroutes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayroutes.yaml new file mode 100644 index 000000000..ef006e9cb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgatewayroutes.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshgatewayroutes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshGatewayRoute + listKind: MeshGatewayRouteList + plural: meshgatewayroutes + singular: meshgatewayroute + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma MeshGatewayRoute resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgateways.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgateways.yaml new file mode 100644 index 000000000..20ff66677 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshgateways.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshgateways.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshGateway + listKind: MeshGatewayList + plural: meshgateways + singular: meshgateway + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma MeshGateway resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhealthchecks.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhealthchecks.yaml new file mode 100644 index 000000000..d1a3a49f9 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhealthchecks.yaml @@ -0,0 +1,382 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshhealthchecks.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshHealthCheck + listKind: MeshHealthCheckList + plural: meshhealthchecks + singular: meshhealthcheck + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshHealthCheck resource. + properties: + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between the consumed services and + corresponding configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + alwaysLogHealthCheckFailures: + description: |- + If set to true, health check failure events will always be logged. If set + to false, only the initial health check failure event will be logged. The + default value is false. + type: boolean + eventLogPath: + description: |- + Specifies the path to the file where Envoy can log health check events. + If empty, no event log will be written. + type: string + failTrafficOnPanic: + description: |- + If set to true, Envoy will not consider any hosts when the cluster is in + 'panic mode'. Instead, the cluster will fail all requests as if all hosts + are unhealthy. This can help avoid potentially overwhelming a failing + service. + type: boolean + grpc: + description: |- + GrpcHealthCheck defines gRPC configuration which will instruct the service + the health check will be made for is a gRPC service. + properties: + authority: + description: |- + The value of the :authority header in the gRPC health check request, + by default name of the cluster this health check is associated with + type: string + disabled: + description: If true the GrpcHealthCheck is disabled + type: boolean + serviceName: + description: Service name parameter which will be sent + to gRPC service + type: string + type: object + healthyPanicThreshold: + anyOf: + - type: integer + - type: string + description: |- + Allows to configure panic threshold for Envoy cluster. If not specified, + the default is 50%. To disable panic mode, set to 0%. + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + healthyThreshold: + default: 1 + description: Number of consecutive healthy checks before + considering a host healthy. + format: int32 + type: integer + http: + description: |- + HttpHealthCheck defines HTTP configuration which will instruct the service + the health check will be made for is an HTTP service. + properties: + disabled: + description: If true the HttpHealthCheck is disabled + type: boolean + expectedStatuses: + description: List of HTTP response statuses which are + considered healthy + items: + format: int32 + type: integer + type: array + path: + default: / + description: |- + The HTTP path which will be requested during the health check + (ie. /health) + type: string + requestHeadersToAdd: + description: |- + The list of HTTP headers which should be added to each health check + request + properties: + add: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + set: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + 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: object + initialJitter: + description: |- + If specified, Envoy will start health checking after a random time in + ms between 0 and initialJitter. This only applies to the first health + check. + type: string + interval: + default: 1m + description: Interval between consecutive health checks. + type: string + intervalJitter: + description: |- + If specified, during every interval Envoy will add IntervalJitter to the + wait time. + type: string + intervalJitterPercent: + description: |- + If specified, during every interval Envoy will add IntervalJitter * + IntervalJitterPercent / 100 to the wait time. If IntervalJitter and + IntervalJitterPercent are both set, both of them will be used to + increase the wait time. + format: int32 + type: integer + noTrafficInterval: + description: |- + The "no traffic interval" is a special health check interval that is used + when a cluster has never had traffic routed to it. This lower interval + allows cluster information to be kept up to date, without sending a + potentially large amount of active health checking traffic for no reason. + Once a cluster has been used for traffic routing, Envoy will shift back + to using the standard health check interval that is defined. Note that + this interval takes precedence over any other. The default value for "no + traffic interval" is 60 seconds. + type: string + reuseConnection: + description: Reuse health check connection between health + checks. Default is true. + type: boolean + tcp: + description: |- + TcpHealthCheck defines configuration for specifying bytes to send and + expected response during the health check + properties: + disabled: + description: If true the TcpHealthCheck is disabled + type: boolean + receive: + description: |- + List of Base64 encoded blocks of strings expected as a response. When checking the response, + "fuzzy" matching is performed such that each block must be found, and + in the order specified, but not necessarily contiguous. + If not provided or empty, checks will be performed as "connect only" and be marked as successful when TCP connection is successfully established. + items: + type: string + type: array + send: + description: Base64 encoded content of the message which + will be sent during the health check to the target + type: string + type: object + timeout: + default: 15s + description: Maximum time to wait for a health check response. + type: string + unhealthyThreshold: + default: 5 + description: |- + Number of consecutive unhealthy checks before considering a host + unhealthy. + format: int32 + type: integer + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhttproutes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhttproutes.yaml new file mode 100644 index 000000000..14f8974b1 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshhttproutes.yaml @@ -0,0 +1,668 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshhttproutes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshHTTPRoute + listKind: MeshHTTPRouteList + plural: meshhttproutes + singular: meshhttproute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshHTTPRoute resource. + properties: + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To matches destination services of requests and holds + configuration. + items: + properties: + hostnames: + description: |- + Hostnames is only valid when targeting MeshGateway and limits the + effects of the rules to requests to this hostname. + Given hostnames must intersect with the hostname of the listeners the + route attaches to. + items: + type: string + type: array + rules: + description: |- + Rules contains the routing rules applies to a combination of top-level + targetRef and the targetRef in this entry. + items: + properties: + default: + description: |- + Default holds routing rules that can be merged with rules from other + policies. + properties: + backendRefs: + items: + description: BackendRef defines where to forward + traffic. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use + to identify cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + port: + description: Port is only supported when this + ref refers to a real MeshService object + format: int32 + type: integer + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + weight: + default: 1 + minimum: 0 + type: integer + type: object + type: array + filters: + items: + properties: + requestHeaderModifier: + description: |- + Only one action is supported per header name. + Configuration to set or add multiple values for a header must use RFC 7230 + header value formatting, separating each value with a comma. + properties: + add: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + items: + type: string + maxItems: 16 + type: array + set: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + 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: + properties: + backendRef: + description: BackendRef defines where to + forward traffic. + properties: + kind: + description: Kind of the referenced + resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future + use to identify cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + port: + description: Port is only supported + when this ref refers to a real MeshService + object + format: int32 + type: integer + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + weight: + default: 1 + minimum: 0 + type: integer + type: object + percentage: + anyOf: + - type: integer + - type: string + description: |- + Percentage of requests to mirror. If not specified, all requests + to the target cluster will be mirrored. + x-kubernetes-int-or-string: true + required: + - backendRef + type: object + requestRedirect: + properties: + hostname: + description: |- + PreciseHostname is the fully qualified domain name of a network host. This + matches the RFC 1123 definition of a hostname with 1 notable exception that + numeric IP addresses are not allowed. + + 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 + 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. + properties: + replaceFullPath: + type: string + replacePrefixMatch: + type: string + type: + 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. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + scheme: + enum: + - http + - https + type: string + statusCode: + default: 302 + description: StatusCode is the HTTP status + code to be used in response. + enum: + - 301 + - 302 + - 303 + - 307 + - 308 + type: integer + type: object + responseHeaderModifier: + description: |- + Only one action is supported per header name. + Configuration to set or add multiple values for a header must use RFC 7230 + header value formatting, separating each value with a comma. + properties: + add: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + remove: + items: + type: string + maxItems: 16 + type: array + set: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + 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: + enum: + - RequestHeaderModifier + - ResponseHeaderModifier + - RequestRedirect + - URLRewrite + - RequestMirror + type: string + urlRewrite: + properties: + hostToBackendHostname: + description: |- + HostToBackendHostname rewrites the hostname to the hostname of the + upstream host. This option is only available when targeting MeshGateways. + type: boolean + hostname: + description: Hostname is the value to be + used to replace the host header value + during forwarding. + 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. + properties: + replaceFullPath: + type: string + replacePrefixMatch: + type: string + type: + enum: + - ReplaceFullPath + - ReplacePrefixMatch + type: string + required: + - type + type: object + type: object + required: + - type + type: object + type: array + type: object + matches: + description: |- + Matches describes how to match HTTP requests this rule should be applied + to. + items: + properties: + headers: + items: + description: |- + HeaderMatch describes how to select an HTTP route by matching HTTP request + headers. + properties: + name: + description: |- + Name is the name of the HTTP Header to be matched. Name MUST be lower case + as they will be handled with case insensitivity (See https://tools.ietf.org/html/rfc7230#section-3.2). + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: Type specifies how to match against + the value of the header. + enum: + - Exact + - Present + - RegularExpression + - Absent + - Prefix + type: string + value: + description: Value is the value of HTTP Header + to be matched. + type: string + required: + - name + type: object + type: array + method: + enum: + - CONNECT + - DELETE + - GET + - HEAD + - OPTIONS + - PATCH + - POST + - PUT + - TRACE + type: string + path: + properties: + type: + enum: + - Exact + - PathPrefix + - RegularExpression + type: string + value: + description: |- + Exact or prefix matches must be an absolute path. A prefix matches only + if separated by a slash or the entire path. + minLength: 1 + type: string + required: + - type + - value + type: object + queryParams: + description: |- + QueryParams matches based on HTTP URL query parameters. Multiple matches + are ANDed together such that all listed matches must succeed. + items: + properties: + name: + minLength: 1 + type: string + type: + enum: + - Exact + - RegularExpression + type: string + value: + type: string + required: + - name + - type + - value + type: object + type: array + type: object + minItems: 1 + type: array + required: + - default + - matches + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + request destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshinsights.yaml new file mode 100644 index 000000000..93b570048 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshinsights.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshInsight + listKind: MeshInsightList + plural: meshinsights + singular: meshinsight + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma MeshInsight resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshloadbalancingstrategies.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshloadbalancingstrategies.yaml new file mode 100644 index 000000000..8fe3d6634 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshloadbalancingstrategies.yaml @@ -0,0 +1,572 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshloadbalancingstrategies.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshLoadBalancingStrategy + listKind: MeshLoadBalancingStrategyList + plural: meshloadbalancingstrategies + singular: meshloadbalancingstrategy + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshLoadBalancingStrategy + resource. + properties: + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between the consumed services and + corresponding configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + loadBalancer: + description: LoadBalancer allows to specify load balancing + algorithm. + properties: + leastRequest: + description: |- + LeastRequest selects N random available hosts as specified in 'choiceCount' (2 by default) + and picks the host which has the fewest active requests + properties: + activeRequestBias: + anyOf: + - type: integer + - type: string + description: |- + ActiveRequestBias refers to dynamic weights applied when hosts have varying load + balancing weights. A higher value here aggressively reduces the weight of endpoints + that are currently handling active requests. In essence, the higher the ActiveRequestBias + value, the more forcefully it reduces the load balancing weight of endpoints that are + actively serving requests. + x-kubernetes-int-or-string: true + choiceCount: + description: |- + ChoiceCount is the number of random healthy hosts from which the host with + the fewest active requests will be chosen. Defaults to 2 so that Envoy performs + two-choice selection if the field is not set. + format: int32 + minimum: 2 + type: integer + type: object + maglev: + description: |- + Maglev implements consistent hashing to upstream hosts. Maglev can be used as + a drop in replacement for the ring hash load balancer any place in which + consistent hashing is desired. + properties: + hashPolicies: + description: |- + HashPolicies specify a list of request/connection properties that are used to calculate a hash. + These hash policies are executed in the specified order. If a hash policy has the “terminal” attribute + set to true, and there is already a hash generated, the hash is returned immediately, + ignoring the rest of the hash policy list. + items: + properties: + connection: + properties: + sourceIP: + description: Hash on source IP address. + type: boolean + type: object + cookie: + properties: + name: + description: The name of the cookie that + will be used to obtain the hash key. + minLength: 1 + type: string + path: + description: The name of the path for + the cookie. + type: string + ttl: + description: If specified, a cookie with + the TTL will be generated if the cookie + is not present. + type: string + required: + - name + type: object + filterState: + properties: + key: + description: |- + The name of the Object in the per-request filterState, which is + an Envoy::Hashable object. If there is no data associated with the key, + or the stored object is not Envoy::Hashable, no hash will be produced. + minLength: 1 + type: string + required: + - key + type: object + header: + properties: + name: + description: The name of the request header + that will be used to obtain the hash + key. + minLength: 1 + type: string + required: + - name + type: object + queryParameter: + properties: + name: + description: |- + The name of the URL query parameter that will be used to obtain the hash key. + If the parameter is not present, no hash will be produced. Query parameter names + are case-sensitive. + minLength: 1 + type: string + required: + - name + type: object + terminal: + description: |- + Terminal is a flag that short-circuits the hash computing. This field provides + a ‘fallback’ style of configuration: “if a terminal policy doesn’t work, fallback + to rest of the policy list”, it saves time when the terminal policy works. + If true, and there is already a hash computed, ignore rest of the list of hash polices. + type: boolean + type: + enum: + - Header + - Cookie + - SourceIP + - QueryParameter + - FilterState + type: string + required: + - type + type: object + type: array + tableSize: + description: |- + The table size for Maglev hashing. Maglev aims for “minimal disruption” + rather than an absolute guarantee. Minimal disruption means that when + the set of upstream hosts change, a connection will likely be sent + to the same upstream as it was before. Increasing the table size reduces + the amount of disruption. The table size must be prime number limited to 5000011. + If it is not specified, the default is 65537. + format: int32 + maximum: 5000011 + minimum: 1 + type: integer + type: object + random: + description: |- + Random selects a random available host. The random load balancer generally + performs better than round-robin if no health checking policy is configured. + Random selection avoids bias towards the host in the set that comes after a failed host. + type: object + ringHash: + description: |- + RingHash implements consistent hashing to upstream hosts. Each host is mapped + onto a circle (the “ring”) by hashing its address; each request is then routed + to a host by hashing some property of the request, and finding the nearest + corresponding host clockwise around the ring. + properties: + hashFunction: + description: |- + HashFunction is a function used to hash hosts onto the ketama ring. + The value defaults to XX_HASH. Available values – XX_HASH, MURMUR_HASH_2. + enum: + - XXHash + - MurmurHash2 + type: string + hashPolicies: + description: |- + HashPolicies specify a list of request/connection properties that are used to calculate a hash. + These hash policies are executed in the specified order. If a hash policy has the “terminal” attribute + set to true, and there is already a hash generated, the hash is returned immediately, + ignoring the rest of the hash policy list. + items: + properties: + connection: + properties: + sourceIP: + description: Hash on source IP address. + type: boolean + type: object + cookie: + properties: + name: + description: The name of the cookie that + will be used to obtain the hash key. + minLength: 1 + type: string + path: + description: The name of the path for + the cookie. + type: string + ttl: + description: If specified, a cookie with + the TTL will be generated if the cookie + is not present. + type: string + required: + - name + type: object + filterState: + properties: + key: + description: |- + The name of the Object in the per-request filterState, which is + an Envoy::Hashable object. If there is no data associated with the key, + or the stored object is not Envoy::Hashable, no hash will be produced. + minLength: 1 + type: string + required: + - key + type: object + header: + properties: + name: + description: The name of the request header + that will be used to obtain the hash + key. + minLength: 1 + type: string + required: + - name + type: object + queryParameter: + properties: + name: + description: |- + The name of the URL query parameter that will be used to obtain the hash key. + If the parameter is not present, no hash will be produced. Query parameter names + are case-sensitive. + minLength: 1 + type: string + required: + - name + type: object + terminal: + description: |- + Terminal is a flag that short-circuits the hash computing. This field provides + a ‘fallback’ style of configuration: “if a terminal policy doesn’t work, fallback + to rest of the policy list”, it saves time when the terminal policy works. + If true, and there is already a hash computed, ignore rest of the list of hash polices. + type: boolean + type: + enum: + - Header + - Cookie + - SourceIP + - QueryParameter + - FilterState + type: string + required: + - type + type: object + type: array + maxRingSize: + description: |- + Maximum hash ring size. Defaults to 8M entries, and limited to 8M entries, + but can be lowered to further constrain resource use. + format: int32 + maximum: 8000000 + minimum: 1 + type: integer + minRingSize: + description: |- + Minimum hash ring size. The larger the ring is (that is, + the more hashes there are for each provided host) the better the request distribution + will reflect the desired weights. Defaults to 1024 entries, and limited to 8M entries. + format: int32 + maximum: 8000000 + minimum: 1 + type: integer + type: object + roundRobin: + description: |- + RoundRobin is a load balancing algorithm that distributes requests + across available upstream hosts in round-robin order. + type: object + type: + enum: + - RoundRobin + - LeastRequest + - RingHash + - Random + - Maglev + type: string + required: + - type + type: object + localityAwareness: + description: LocalityAwareness contains configuration for + locality aware load balancing. + properties: + crossZone: + description: |- + CrossZone defines locality aware load balancing priorities when dataplane proxies inside local zone + are unavailable + properties: + failover: + description: Failover defines list of load balancing + rules in order of priority + items: + properties: + from: + description: From defines the list of zones + to which the rule applies + properties: + zones: + items: + type: string + type: array + required: + - zones + type: object + to: + description: To defines to which zones the + traffic should be load balanced + properties: + type: + description: Type defines how target zones + will be picked from available zones + enum: + - None + - Only + - Any + - AnyExcept + type: string + zones: + items: + type: string + type: array + required: + - type + type: object + required: + - to + type: object + type: array + failoverThreshold: + description: |- + FailoverThreshold defines the percentage of live destination dataplane proxies below which load balancing to the + next priority starts. + Example: If you configure failoverThreshold to 70, and you have deployed 10 destination dataplane proxies. + Load balancing to next priority will start when number of live destination dataplane proxies drops below 7. + Default 50 + properties: + percentage: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - percentage + type: object + type: object + disabled: + description: |- + Disabled allows to disable locality-aware load balancing. + When disabled requests are distributed across all endpoints regardless of locality. + type: boolean + localZone: + description: LocalZone defines locality aware load balancing + priorities between dataplane proxies inside a zone + properties: + affinityTags: + description: AffinityTags list of tags for local + zone load balancing. + items: + properties: + key: + description: Key defines tag for which affinity + is configured + type: string + weight: + description: |- + Weight of the tag used for load balancing. The bigger the weight the bigger the priority. + Percentage of local traffic load balanced to tag is computed by dividing weight by sum of weights from all tags. + For example with two affinity tags first with weight 80 and second with weight 20, + then 80% of traffic will be redirected to the first tag, and 20% of traffic will be redirected to second one. + Setting weights is not mandatory. When weights are not set control plane will compute default weight based on list order. + Default: If you do not specify weight we will adjust them so that 90% traffic goes to first tag, 9% to next, and 1% to third and so on. + format: int32 + type: integer + required: + - key + type: object + type: array + type: object + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmetrics.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmetrics.yaml new file mode 100644 index 000000000..d244c2e04 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmetrics.yaml @@ -0,0 +1,292 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshmetrics.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshMetric + listKind: MeshMetricList + plural: meshmetrics + singular: meshmetric + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshMetric resource. + properties: + default: + description: MeshMetric configuration. + properties: + applications: + description: Applications is a list of application that Dataplane + Proxy will scrape + items: + properties: + address: + description: Address on which an application listens. + type: string + name: + description: Name of the application to scrape + type: string + path: + default: /metrics/prometheus + description: Path on which an application expose HTTP endpoint + with metrics. + type: string + port: + description: Port on which an application expose HTTP endpoint + with metrics. + format: int32 + type: integer + required: + - port + type: object + type: array + backends: + description: Backends list that will be used to collect metrics. + items: + properties: + openTelemetry: + description: OpenTelemetry backend configuration + properties: + endpoint: + description: Endpoint for OpenTelemetry collector + type: string + refreshInterval: + description: RefreshInterval defines how frequent metrics + should be pushed to collector + type: string + required: + - endpoint + type: object + prometheus: + description: Prometheus backend configuration. + properties: + clientId: + description: ClientId of the Prometheus backend. Needed + when using MADS for DP discovery. + type: string + path: + default: /metrics + description: Path on which a dataplane should expose + HTTP endpoint with Prometheus metrics. + type: string + port: + default: 5670 + description: Port on which a dataplane should expose + HTTP endpoint with Prometheus metrics. + format: int32 + type: integer + tls: + description: Configuration of TLS for prometheus listener. + properties: + mode: + default: Disabled + description: Configuration of TLS for Prometheus + listener. + enum: + - Disabled + - ProvidedTLS + - ActiveMTLSBackend + type: string + required: + - mode + type: object + required: + - path + - port + type: object + type: + description: Type of the backend that will be used to collect + metrics. At the moment only Prometheus backend is available. + enum: + - Prometheus + - OpenTelemetry + type: string + required: + - type + type: object + type: array + sidecar: + description: Sidecar metrics collection configuration + properties: + includeUnused: + default: false + description: |- + IncludeUnused if false will scrape only metrics that has been by sidecar (counters incremented + at least once, gauges changed at least once, and histograms added to at + least once). If true will scrape all metrics (even the ones with zeros). + type: boolean + profiles: + description: Profiles allows to customize which metrics are + published. + properties: + appendProfiles: + description: AppendProfiles allows to combine the metrics + from multiple predefined profiles. + items: + properties: + name: + description: 'Name of the predefined profile, one + of: all, basic, none' + enum: + - All + - Basic + - None + type: string + required: + - name + type: object + type: array + exclude: + description: |- + Exclude makes it possible to exclude groups of metrics from a resulting profile. + Exclude is subordinate to Include. + items: + properties: + match: + description: Match is the value used to match using + particular Type + type: string + type: + description: 'Type defined the type of selector, + one of: prefix, regex, exact' + enum: + - Prefix + - Regex + - Exact + - Contains + type: string + required: + - match + - type + type: object + type: array + include: + description: |- + Include makes it possible to include additional metrics in a selected profiles. + Include takes precedence over Exclude. + items: + properties: + match: + description: Match is the value used to match using + particular Type + type: string + type: + description: 'Type defined the type of selector, + one of: prefix, regex, exact' + enum: + - Prefix + - Regex + - Exact + - Contains + type: string + required: + - match + - type + type: object + type: array + type: object + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in-place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmultizoneservices.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmultizoneservices.yaml new file mode 100644 index 000000000..4772b0cfb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshmultizoneservices.yaml @@ -0,0 +1,199 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshmultizoneservices.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshMultiZoneService + listKind: MeshMultiZoneServiceList + plural: meshmultizoneservices + singular: meshmultizoneservice + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.addresses[0].hostname + name: Hostname + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshMultiZoneService + resource. + properties: + ports: + description: Ports is a list of ports from selected MeshServices + items: + properties: + appProtocol: + default: tcp + description: Protocol identifies a protocol supported by a service. + type: string + name: + type: string + port: + format: int32 + type: integer + required: + - port + type: object + minItems: 1 + type: array + selector: + description: Selector is a way to select multiple MeshServices + properties: + meshService: + description: MeshService selects MeshServices + properties: + matchLabels: + additionalProperties: + type: string + description: MatchLabels matches multiple MeshServices by + labels + type: object + required: + - matchLabels + type: object + required: + - meshService + type: object + required: + - selector + type: object + status: + description: Status is the current status of the Kuma MeshMultiZoneService + resource. + properties: + addresses: + description: Addresses is a list of addresses generated by HostnameGenerator + items: + properties: + hostname: + type: string + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + origin: + type: string + type: object + type: array + hostnameGenerators: + description: Status of hostnames generator applied on this resource + items: + properties: + conditions: + description: Conditions is an array of hostname generator conditions. + items: + properties: + message: + description: |- + message is a human readable message indicating details about the transition. + This may be an empty string. + maxLength: 32768 + 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 + 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. + 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: + - message + - reason + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + required: + - hostnameGeneratorRef + type: object + type: array + meshServices: + description: MeshServices is a list of matched MeshServices + items: + properties: + mesh: + type: string + name: + description: Name is a core name of MeshService + type: string + namespace: + type: string + zone: + type: string + required: + - mesh + - name + - namespace + - zone + type: object + type: array + vips: + description: VIPs is a list of assigned Kuma VIPs. + items: + properties: + ip: + type: string + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshpassthroughs.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshpassthroughs.yaml new file mode 100644 index 000000000..9f5822b55 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshpassthroughs.yaml @@ -0,0 +1,164 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshpassthroughs.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshPassthrough + listKind: MeshPassthroughList + plural: meshpassthroughs + singular: meshpassthrough + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshPassthrough resource. + properties: + default: + description: MeshPassthrough configuration. + properties: + appendMatch: + description: AppendMatch is a list of destinations that should + be allowed through the sidecar. + items: + properties: + port: + description: Port defines the port to which a user makes + a request. + type: integer + protocol: + default: tcp + description: 'Protocol defines the communication protocol. + Possible values: `tcp`, `tls`, `grpc`, `http`, `http2`.' + enum: + - tcp + - tls + - grpc + - http + - http2 + type: string + type: + description: Type of the match, one of `Domain`, `IP` or + `CIDR` is available. + enum: + - Domain + - IP + - CIDR + type: string + value: + description: Value for the specified Type. + type: string + type: object + type: array + passthroughMode: + default: None + description: |- + Defines the passthrough behavior. Possible values: `All`, `None`, `Matched` + When `All` or `None` `appendMatch` has no effect. + enum: + - All + - Matched + - None + type: string + type: object + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in-place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshproxypatches.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshproxypatches.yaml new file mode 100644 index 000000000..bf6342d25 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshproxypatches.yaml @@ -0,0 +1,550 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshproxypatches.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshProxyPatch + listKind: MeshProxyPatchList + plural: meshproxypatches + singular: meshproxypatch + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshProxyPatch resource. + properties: + default: + description: |- + Default is a configuration specific to the group of destinations + referenced in 'targetRef'. + properties: + appendModifications: + description: AppendModifications is a list of modifications applied + on the selected proxy. + items: + properties: + cluster: + description: Cluster is a modification of Envoy's Cluster + resource. + properties: + jsonPatches: + description: |- + JsonPatches specifies list of jsonpatches to apply to on Envoy's Cluster + resource + items: + description: JsonPatchBlock is one json patch operation + block. + properties: + from: + description: From is a jsonpatch from string, + used by move and copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: Value must be a valid json value + used by replace and add operations. + x-kubernetes-preserve-unknown-fields: true + required: + - op + - path + type: object + type: array + match: + description: Match is a set of conditions that have + to be matched for modification operation to happen. + properties: + name: + description: Name of the cluster to match. + type: string + origin: + description: |- + Origin is the name of the component or plugin that generated the resource. + + Here is the list of well-known origins: + inbound - resources generated for handling incoming traffic. + outbound - resources generated for handling outgoing traffic. + transparent - resources generated for transparent proxy functionality. + prometheus - resources generated when Prometheus metrics are enabled. + direct-access - resources generated for Direct Access functionality. + ingress - resources generated for Zone Ingress. + egress - resources generated for Zone Egress. + gateway - resources generated for MeshGateway. + + The list is not complete, because policy plugins can introduce new resources. + For example MeshTrace plugin can create Cluster with "mesh-trace" origin. + type: string + type: object + operation: + description: Operation to execute on matched cluster. + enum: + - Add + - Remove + - Patch + type: string + value: + description: Value of xDS resource in YAML format to + add or patch. + type: string + required: + - operation + type: object + httpFilter: + description: |- + HTTPFilter is a modification of Envoy HTTP Filter + available in HTTP Connection Manager in a Listener resource. + properties: + jsonPatches: + description: |- + JsonPatches specifies list of jsonpatches to apply to on Envoy's + HTTP Filter available in HTTP Connection Manager in a Listener resource. + items: + description: JsonPatchBlock is one json patch operation + block. + properties: + from: + description: From is a jsonpatch from string, + used by move and copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: Value must be a valid json value + used by replace and add operations. + x-kubernetes-preserve-unknown-fields: true + required: + - op + - path + type: object + type: array + match: + description: Match is a set of conditions that have + to be matched for modification operation to happen. + properties: + listenerName: + description: Name of the listener to match. + type: string + listenerTags: + additionalProperties: + type: string + description: Listener tags available in Listener#Metadata#FilterMetadata[io.kuma.tags] + type: object + name: + description: Name of the HTTP filter. For example + "envoy.filters.http.local_ratelimit" + type: string + origin: + description: |- + Origin is the name of the component or plugin that generated the resource. + + Here is the list of well-known origins: + inbound - resources generated for handling incoming traffic. + outbound - resources generated for handling outgoing traffic. + transparent - resources generated for transparent proxy functionality. + prometheus - resources generated when Prometheus metrics are enabled. + direct-access - resources generated for Direct Access functionality. + ingress - resources generated for Zone Ingress. + egress - resources generated for Zone Egress. + gateway - resources generated for MeshGateway. + + The list is not complete, because policy plugins can introduce new resources. + For example MeshTrace plugin can create Cluster with "mesh-trace" origin. + type: string + type: object + operation: + description: Operation to execute on matched listener. + enum: + - Remove + - Patch + - AddFirst + - AddBefore + - AddAfter + - AddLast + type: string + value: + description: Value of xDS resource in YAML format to + add or patch. + type: string + required: + - operation + type: object + listener: + description: Listener is a modification of Envoy's Listener + resource. + properties: + jsonPatches: + description: |- + JsonPatches specifies list of jsonpatches to apply to on Envoy's Listener + resource + items: + description: JsonPatchBlock is one json patch operation + block. + properties: + from: + description: From is a jsonpatch from string, + used by move and copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: Value must be a valid json value + used by replace and add operations. + x-kubernetes-preserve-unknown-fields: true + required: + - op + - path + type: object + type: array + match: + description: Match is a set of conditions that have + to be matched for modification operation to happen. + properties: + name: + description: Name of the listener to match. + type: string + origin: + description: |- + Origin is the name of the component or plugin that generated the resource. + + Here is the list of well-known origins: + inbound - resources generated for handling incoming traffic. + outbound - resources generated for handling outgoing traffic. + transparent - resources generated for transparent proxy functionality. + prometheus - resources generated when Prometheus metrics are enabled. + direct-access - resources generated for Direct Access functionality. + ingress - resources generated for Zone Ingress. + egress - resources generated for Zone Egress. + gateway - resources generated for MeshGateway. + + The list is not complete, because policy plugins can introduce new resources. + For example MeshTrace plugin can create Cluster with "mesh-trace" origin. + type: string + tags: + additionalProperties: + type: string + description: Tags available in Listener#Metadata#FilterMetadata[io.kuma.tags] + type: object + type: object + operation: + description: Operation to execute on matched listener. + enum: + - Add + - Remove + - Patch + type: string + value: + description: Value of xDS resource in YAML format to + add or patch. + type: string + required: + - operation + type: object + networkFilter: + description: NetworkFilter is a modification of Envoy Listener's + filter. + properties: + jsonPatches: + description: |- + JsonPatches specifies list of jsonpatches to apply to on Envoy Listener's + filter. + items: + description: JsonPatchBlock is one json patch operation + block. + properties: + from: + description: From is a jsonpatch from string, + used by move and copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: Value must be a valid json value + used by replace and add operations. + x-kubernetes-preserve-unknown-fields: true + required: + - op + - path + type: object + type: array + match: + description: Match is a set of conditions that have + to be matched for modification operation to happen. + properties: + listenerName: + description: Name of the listener to match. + type: string + listenerTags: + additionalProperties: + type: string + description: Listener tags available in Listener#Metadata#FilterMetadata[io.kuma.tags] + type: object + name: + description: Name of the network filter. For example + "envoy.filters.network.ratelimit" + type: string + origin: + description: |- + Origin is the name of the component or plugin that generated the resource. + + Here is the list of well-known origins: + inbound - resources generated for handling incoming traffic. + outbound - resources generated for handling outgoing traffic. + transparent - resources generated for transparent proxy functionality. + prometheus - resources generated when Prometheus metrics are enabled. + direct-access - resources generated for Direct Access functionality. + ingress - resources generated for Zone Ingress. + egress - resources generated for Zone Egress. + gateway - resources generated for MeshGateway. + + The list is not complete, because policy plugins can introduce new resources. + For example MeshTrace plugin can create Cluster with "mesh-trace" origin. + type: string + type: object + operation: + description: Operation to execute on matched listener. + enum: + - Remove + - Patch + - AddFirst + - AddBefore + - AddAfter + - AddLast + type: string + value: + description: Value of xDS resource in YAML format to + add or patch. + type: string + required: + - operation + type: object + virtualHost: + description: |- + VirtualHost is a modification of Envoy's VirtualHost + referenced in HTTP Connection Manager in a Listener resource. + properties: + jsonPatches: + description: |- + JsonPatches specifies list of jsonpatches to apply to on Envoy's + VirtualHost resource + items: + description: JsonPatchBlock is one json patch operation + block. + properties: + from: + description: From is a jsonpatch from string, + used by move and copy operations. + type: string + op: + description: Op is a jsonpatch operation string. + enum: + - add + - remove + - replace + - move + - copy + type: string + path: + description: Path is a jsonpatch path string. + type: string + value: + description: Value must be a valid json value + used by replace and add operations. + x-kubernetes-preserve-unknown-fields: true + required: + - op + - path + type: object + type: array + match: + description: Match is a set of conditions that have + to be matched for modification operation to happen. + properties: + name: + description: Name of the VirtualHost to match. + type: string + origin: + description: |- + Origin is the name of the component or plugin that generated the resource. + + Here is the list of well-known origins: + inbound - resources generated for handling incoming traffic. + outbound - resources generated for handling outgoing traffic. + transparent - resources generated for transparent proxy functionality. + prometheus - resources generated when Prometheus metrics are enabled. + direct-access - resources generated for Direct Access functionality. + ingress - resources generated for Zone Ingress. + egress - resources generated for Zone Egress. + gateway - resources generated for MeshGateway. + + The list is not complete, because policy plugins can introduce new resources. + For example MeshTrace plugin can create Cluster with "mesh-trace" origin. + type: string + routeConfigurationName: + description: Name of the RouteConfiguration resource + to match. + type: string + type: object + operation: + description: Operation to execute on matched listener. + enum: + - Add + - Remove + - Patch + type: string + value: + description: Value of xDS resource in YAML format to + add or patch. + type: string + required: + - match + - operation + type: object + type: object + type: array + required: + - appendModifications + type: object + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - default + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshratelimits.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshratelimits.yaml new file mode 100644 index 000000000..52424a985 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshratelimits.yaml @@ -0,0 +1,499 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshratelimits.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshRateLimit + listKind: MeshRateLimitList + plural: meshratelimits + singular: meshratelimit + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshRateLimit resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + local: + description: LocalConf defines local http or/and tcp rate + limit configuration + properties: + http: + description: |- + LocalHTTP defines configuration of local HTTP rate limiting + https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/local_rate_limit_filter + properties: + disabled: + description: Define if rate limiting should be disabled. + type: boolean + onRateLimit: + description: Describes the actions to take on a + rate limit event + properties: + headers: + description: The Headers to be added to the + HTTP response on a rate limit event + properties: + add: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + set: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + status: + description: The HTTP status code to be set + on a rate limit event + format: int32 + type: integer + type: object + requestRate: + description: Defines how many requests are allowed + per interval. + properties: + interval: + description: The interval the number of units + is accounted for. + type: string + num: + description: |- + Number of units per interval (depending on usage it can be a number of requests, + or a number of connections). + format: int32 + type: integer + required: + - interval + - num + type: object + type: object + tcp: + description: |- + LocalTCP defines confguration of local TCP rate limiting + https://www.envoyproxy.io/docs/envoy/latest/configuration/listeners/network_filters/local_rate_limit_filter + properties: + connectionRate: + description: Defines how many connections are allowed + per interval. + properties: + interval: + description: The interval the number of units + is accounted for. + type: string + num: + description: |- + Number of units per interval (depending on usage it can be a number of requests, + or a number of connections). + format: int32 + type: integer + required: + - interval + - num + type: object + disabled: + description: |- + Define if rate limiting should be disabled. + Default: false + type: boolean + type: object + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + local: + description: LocalConf defines local http or/and tcp rate + limit configuration + properties: + http: + description: |- + LocalHTTP defines configuration of local HTTP rate limiting + https://www.envoyproxy.io/docs/envoy/latest/configuration/http/http_filters/local_rate_limit_filter + properties: + disabled: + description: Define if rate limiting should be disabled. + type: boolean + onRateLimit: + description: Describes the actions to take on a + rate limit event + properties: + headers: + description: The Headers to be added to the + HTTP response on a rate limit event + properties: + add: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + set: + items: + properties: + name: + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + value: + type: string + required: + - name + - value + type: object + maxItems: 16 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + status: + description: The HTTP status code to be set + on a rate limit event + format: int32 + type: integer + type: object + requestRate: + description: Defines how many requests are allowed + per interval. + properties: + interval: + description: The interval the number of units + is accounted for. + type: string + num: + description: |- + Number of units per interval (depending on usage it can be a number of requests, + or a number of connections). + format: int32 + type: integer + required: + - interval + - num + type: object + type: object + tcp: + description: |- + LocalTCP defines confguration of local TCP rate limiting + https://www.envoyproxy.io/docs/envoy/latest/configuration/listeners/network_filters/local_rate_limit_filter + properties: + connectionRate: + description: Defines how many connections are allowed + per interval. + properties: + interval: + description: The interval the number of units + is accounted for. + type: string + num: + description: |- + Number of units per interval (depending on usage it can be a number of requests, + or a number of connections). + format: int32 + type: integer + required: + - interval + - num + type: object + disabled: + description: |- + Define if rate limiting should be disabled. + Default: false + type: boolean + type: object + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshretries.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshretries.yaml new file mode 100644 index 000000000..f4337a105 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshretries.yaml @@ -0,0 +1,507 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshretries.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshRetry + listKind: MeshRetryList + plural: meshretries + singular: meshretry + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshRetry resource. + properties: + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between the consumed services and + corresponding configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + grpc: + description: GRPC defines a configuration of retries for + GRPC traffic + properties: + backOff: + description: |- + BackOff is a configuration of durations which will be used in an exponential + backoff strategy between retries. + properties: + baseInterval: + default: 25ms + description: |- + BaseInterval is an amount of time which should be taken between retries. + Must be greater than zero. Values less than 1 ms are rounded up to 1 ms. + type: string + maxInterval: + description: |- + MaxInterval is a maximal amount of time which will be taken between retries. + Default is 10 times the "BaseInterval". + type: string + type: object + numRetries: + description: |- + NumRetries is the number of attempts that will be made on failed (and + retriable) requests. If not set, the default value is 1. + format: int32 + type: integer + perTryTimeout: + description: |- + PerTryTimeout is the maximum amount of time each retry attempt can take + before it times out. If not set, the global request timeout for the route + will be used. Setting this value to 0 will disable the per-try timeout. + type: string + rateLimitedBackOff: + description: |- + RateLimitedBackOff is a configuration of backoff which will be used when + the upstream returns one of the headers configured. + properties: + maxInterval: + default: 300s + description: MaxInterval is a maximal amount of + time which will be taken between retries. + type: string + resetHeaders: + description: |- + ResetHeaders specifies the list of headers (like Retry-After or X-RateLimit-Reset) + to match against the response. Headers are tried in order, and matched + case-insensitive. The first header to be parsed successfully is used. + If no headers match the default exponential BackOff is used instead. + items: + properties: + format: + description: The format of the reset header. + enum: + - Seconds + - UnixTimestamp + type: string + name: + description: The Name of the reset header. + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + required: + - format + - name + type: object + type: array + type: object + retryOn: + description: RetryOn is a list of conditions which will + cause a retry. + example: + - Canceled + - DeadlineExceeded + - Internal + - ResourceExhausted + - Unavailable + items: + enum: + - Canceled + - DeadlineExceeded + - Internal + - ResourceExhausted + - Unavailable + type: string + type: array + type: object + http: + description: HTTP defines a configuration of retries for + HTTP traffic + properties: + backOff: + description: |- + BackOff is a configuration of durations which will be used in exponential + backoff strategy between retries. + properties: + baseInterval: + default: 25ms + description: |- + BaseInterval is an amount of time which should be taken between retries. + Must be greater than zero. Values less than 1 ms are rounded up to 1 ms. + type: string + maxInterval: + description: |- + MaxInterval is a maximal amount of time which will be taken between retries. + Default is 10 times the "BaseInterval". + type: string + type: object + hostSelection: + description: |- + HostSelection is a list of predicates that dictate how hosts should be selected + when requests are retried. + items: + properties: + predicate: + description: Type is requested predicate mode. + enum: + - OmitPreviousHosts + - OmitHostsWithTags + - OmitPreviousPriorities + type: string + tags: + additionalProperties: + type: string + description: |- + Tags is a map of metadata to match against for selecting the omitted hosts. Required if Type is + OmitHostsWithTags + type: object + updateFrequency: + default: 2 + description: |- + UpdateFrequency is how often the priority load should be updated based on previously attempted priorities. + Used for OmitPreviousPriorities. + format: int32 + type: integer + required: + - predicate + type: object + type: array + hostSelectionMaxAttempts: + description: |- + HostSelectionMaxAttempts is the maximum number of times host selection will be + reattempted before giving up, at which point the host that was last selected will + be routed to. If unspecified, this will default to retrying once. + format: int64 + type: integer + numRetries: + description: |- + NumRetries is the number of attempts that will be made on failed (and + retriable) requests. If not set, the default value is 1. + format: int32 + type: integer + perTryTimeout: + description: |- + PerTryTimeout is the amount of time after which retry attempt should time out. + If left unspecified, the global route timeout for the request will be used. + Consequently, when using a 5xx based retry policy, a request that times out + will not be retried as the total timeout budget would have been exhausted. + Setting this timeout to 0 will disable it. + type: string + rateLimitedBackOff: + description: |- + RateLimitedBackOff is a configuration of backoff which will be used + when the upstream returns one of the headers configured. + properties: + maxInterval: + default: 300s + description: MaxInterval is a maximal amount of + time which will be taken between retries. + type: string + resetHeaders: + description: |- + ResetHeaders specifies the list of headers (like Retry-After or X-RateLimit-Reset) + to match against the response. Headers are tried in order, and matched + case-insensitive. The first header to be parsed successfully is used. + If no headers match the default exponential BackOff is used instead. + items: + properties: + format: + description: The format of the reset header. + enum: + - Seconds + - UnixTimestamp + type: string + name: + description: The Name of the reset header. + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + required: + - format + - name + type: object + type: array + type: object + retriableRequestHeaders: + description: |- + RetriableRequestHeaders is an HTTP headers which must be present in the request + for retries to be attempted. + items: + description: |- + HeaderMatch describes how to select an HTTP route by matching HTTP request + headers. + properties: + name: + description: |- + Name is the name of the HTTP Header to be matched. Name MUST be lower case + as they will be handled with case insensitivity (See https://tools.ietf.org/html/rfc7230#section-3.2). + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: Type specifies how to match against + the value of the header. + enum: + - Exact + - Present + - RegularExpression + - Absent + - Prefix + type: string + value: + description: Value is the value of HTTP Header + to be matched. + type: string + required: + - name + type: object + type: array + retriableResponseHeaders: + description: |- + RetriableResponseHeaders is an HTTP response headers that trigger a retry + if present in the response. A retry will be triggered if any of the header + matches the upstream response headers. + items: + description: |- + HeaderMatch describes how to select an HTTP route by matching HTTP request + headers. + properties: + name: + description: |- + Name is the name of the HTTP Header to be matched. Name MUST be lower case + as they will be handled with case insensitivity (See https://tools.ietf.org/html/rfc7230#section-3.2). + maxLength: 256 + minLength: 1 + pattern: ^[a-z0-9!#$%&'*+\-.^_\x60|~]+$ + type: string + type: + default: Exact + description: Type specifies how to match against + the value of the header. + enum: + - Exact + - Present + - RegularExpression + - Absent + - Prefix + type: string + value: + description: Value is the value of HTTP Header + to be matched. + type: string + required: + - name + type: object + type: array + retryOn: + description: |- + RetryOn is a list of conditions which will cause a retry. Available values are: + [5XX, GatewayError, Reset, Retriable4xx, ConnectFailure, EnvoyRatelimited, + RefusedStream, Http3PostConnectFailure, HttpMethodConnect, HttpMethodDelete, + HttpMethodGet, HttpMethodHead, HttpMethodOptions, HttpMethodPatch, + HttpMethodPost, HttpMethodPut, HttpMethodTrace]. + Also, any HTTP status code (500, 503, etc.). + example: + - 5XX + - GatewayError + - Reset + - Retriable4xx + - ConnectFailure + - EnvoyRatelimited + - RefusedStream + - Http3PostConnectFailure + - HttpMethodConnect + - HttpMethodDelete + - HttpMethodGet + - HttpMethodHead + - HttpMethodOptions + - HttpMethodPatch + - HttpMethodPost + - HttpMethodPut + - HttpMethodTrace + - "500" + - "503" + items: + type: string + type: array + type: object + tcp: + description: TCP defines a configuration of retries for + TCP traffic + properties: + maxConnectAttempt: + description: |- + MaxConnectAttempt is a maximal amount of TCP connection attempts + which will be made before giving up + format: int32 + type: integer + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshservices.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshservices.yaml new file mode 100644 index 000000000..5ac9cf40b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshservices.yaml @@ -0,0 +1,218 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshservices.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshService + listKind: MeshServiceList + plural: meshservices + singular: meshservice + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.addresses[0].hostname + name: Hostname + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshService resource. + properties: + identities: + items: + properties: + type: + enum: + - ServiceTag + type: string + value: + type: string + required: + - type + - value + type: object + type: array + ports: + items: + properties: + appProtocol: + default: tcp + description: Protocol identifies a protocol supported by a service. + type: string + name: + type: string + port: + format: int32 + type: integer + targetPort: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + required: + - port + type: object + type: array + x-kubernetes-list-map-keys: + - port + - appProtocol + x-kubernetes-list-type: map + selector: + properties: + dataplaneRef: + properties: + name: + type: string + type: object + dataplaneTags: + additionalProperties: + type: string + type: object + type: object + state: + description: |- + State of MeshService. Available if there is at least one healthy endpoint. Otherwise, Unavailable. + It's used for cross zone communication to check if we should send traffic to it, when MeshService is aggregated into MeshMultiZoneService. + enum: + - Available + - Unavailable + type: string + type: object + status: + description: Status is the current status of the Kuma MeshService resource. + properties: + addresses: + items: + properties: + hostname: + type: string + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + origin: + type: string + type: object + type: array + dataplaneProxies: + description: Data plane proxies statistics selected by this MeshService. + properties: + connected: + description: Number of data plane proxies connected to the zone + control plane + type: integer + healthy: + description: Number of data plane proxies with all healthy inbounds + selected by this MeshService. + type: integer + total: + description: Total number of data plane proxies. + type: integer + type: object + hostnameGenerators: + items: + properties: + conditions: + description: Conditions is an array of hostname generator conditions. + items: + properties: + message: + description: |- + message is a human readable message indicating details about the transition. + This may be an empty string. + maxLength: 32768 + 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 + 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. + 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: + - message + - reason + - status + - type + type: object + type: array + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + hostnameGeneratorRef: + properties: + coreName: + type: string + required: + - coreName + type: object + required: + - hostnameGeneratorRef + type: object + type: array + tls: + properties: + status: + enum: + - Ready + - NotReady + type: string + type: object + vips: + items: + properties: + ip: + type: string + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtcproutes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtcproutes.yaml new file mode 100644 index 000000000..9d1d0ad7e --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtcproutes.yaml @@ -0,0 +1,282 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshtcproutes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshTCPRoute + listKind: MeshTCPRouteList + plural: meshtcproutes + singular: meshtcproute + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshTCPRoute resource. + properties: + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in-place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: |- + To list makes a match between the consumed services and corresponding + configurations + items: + properties: + rules: + description: |- + Rules contains the routing rules applies to a combination of top-level + targetRef and the targetRef in this entry. + items: + properties: + default: + description: |- + Default holds routing rules that can be merged with rules from other + policies. + properties: + backendRefs: + items: + description: BackendRef defines where to forward + traffic. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use + to identify cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + port: + description: Port is only supported when this + ref refers to a real MeshService object + format: int32 + type: integer + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + weight: + default: 1 + minimum: 0 + type: integer + type: object + minItems: 1 + type: array + required: + - backendRefs + type: object + required: + - default + type: object + maxItems: 1 + type: array + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + minItems: 1 + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtimeouts.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtimeouts.yaml new file mode 100644 index 000000000..330873a94 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtimeouts.yaml @@ -0,0 +1,363 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshtimeouts.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshTimeout + listKind: MeshTimeoutList + plural: meshtimeouts + singular: meshtimeout + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshTimeout resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + connectionTimeout: + description: |- + ConnectionTimeout specifies the amount of time proxy will wait for an TCP connection to be established. + Default value is 5 seconds. Cannot be set to 0. + type: string + http: + description: Http provides configuration for HTTP specific + timeouts + properties: + maxConnectionDuration: + description: |- + MaxConnectionDuration is the time after which a connection will be drained and/or closed, + starting from when it was first established. Setting this timeout to 0 will disable it. + Disabled by default. + type: string + maxStreamDuration: + description: |- + MaxStreamDuration is the maximum time that a stream’s lifetime will span. + Setting this timeout to 0 will disable it. Disabled by default. + type: string + requestHeadersTimeout: + description: |- + RequestHeadersTimeout The amount of time that proxy will wait for the request headers to be received. The timer is + activated when the first byte of the headers is received, and is disarmed when the last byte of + the headers has been received. If not specified or set to 0, this timeout is disabled. + Disabled by default. + type: string + requestTimeout: + description: |- + RequestTimeout The amount of time that proxy will wait for the entire request to be received. + The timer is activated when the request is initiated, and is disarmed when the last byte of the request is sent, + OR when the response is initiated. Setting this timeout to 0 will disable it. + Default is 15s. + type: string + streamIdleTimeout: + description: |- + StreamIdleTimeout is the amount of time that proxy will allow a stream to exist with no activity. + Setting this timeout to 0 will disable it. Default is 30m + type: string + type: object + idleTimeout: + description: |- + IdleTimeout is defined as the period in which there are no bytes sent or received on connection + Setting this timeout to 0 will disable it. Be cautious when disabling it because + it can lead to connection leaking. Default value is 1h. + type: string + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + to: + description: To list makes a match between the consumed services and + corresponding configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of destinations referenced in + 'targetRef' + properties: + connectionTimeout: + description: |- + ConnectionTimeout specifies the amount of time proxy will wait for an TCP connection to be established. + Default value is 5 seconds. Cannot be set to 0. + type: string + http: + description: Http provides configuration for HTTP specific + timeouts + properties: + maxConnectionDuration: + description: |- + MaxConnectionDuration is the time after which a connection will be drained and/or closed, + starting from when it was first established. Setting this timeout to 0 will disable it. + Disabled by default. + type: string + maxStreamDuration: + description: |- + MaxStreamDuration is the maximum time that a stream’s lifetime will span. + Setting this timeout to 0 will disable it. Disabled by default. + type: string + requestHeadersTimeout: + description: |- + RequestHeadersTimeout The amount of time that proxy will wait for the request headers to be received. The timer is + activated when the first byte of the headers is received, and is disarmed when the last byte of + the headers has been received. If not specified or set to 0, this timeout is disabled. + Disabled by default. + type: string + requestTimeout: + description: |- + RequestTimeout The amount of time that proxy will wait for the entire request to be received. + The timer is activated when the request is initiated, and is disarmed when the last byte of the request is sent, + OR when the response is initiated. Setting this timeout to 0 will disable it. + Default is 15s. + type: string + streamIdleTimeout: + description: |- + StreamIdleTimeout is the amount of time that proxy will allow a stream to exist with no activity. + Setting this timeout to 0 will disable it. Default is 30m + type: string + type: object + idleTimeout: + description: |- + IdleTimeout is defined as the period in which there are no bytes sent or received on connection + Setting this timeout to 0 will disable it. Be cautious when disabling it because + it can lead to connection leaking. Default value is 1h. + type: string + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + destinations. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtlses.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtlses.yaml new file mode 100644 index 000000000..4ddbfffcb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtlses.yaml @@ -0,0 +1,239 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshtlses.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshTLS + listKind: MeshTLSList + plural: meshtlses + singular: meshtls + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshTLS resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + mode: + description: Mode defines the behavior of inbound listeners + with regard to traffic encryption. + enum: + - Permissive + - Strict + type: string + tlsCiphers: + description: TlsCiphers section for providing ciphers specification. + items: + enum: + - ECDHE-ECDSA-AES128-GCM-SHA256 + - ECDHE-ECDSA-AES256-GCM-SHA384 + - ECDHE-ECDSA-CHACHA20-POLY1305 + - ECDHE-RSA-AES128-GCM-SHA256 + - ECDHE-RSA-AES256-GCM-SHA384 + - ECDHE-RSA-CHACHA20-POLY1305 + type: string + type: array + tlsVersion: + description: Version section for providing version specification. + properties: + max: + default: TLSAuto + description: Max defines maximum supported version. + One of `TLSAuto`, `TLS10`, `TLS11`, `TLS12`, `TLS13`. + enum: + - TLSAuto + - TLS10 + - TLS11 + - TLS12 + - TLS13 + type: string + min: + default: TLSAuto + description: Min defines minimum supported version. + One of `TLSAuto`, `TLS10`, `TLS11`, `TLS12`, `TLS13`. + enum: + - TLSAuto + - TLS10 + - TLS11 + - TLS12 + - TLS13 + type: string + type: object + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined in-place. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtraces.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtraces.yaml new file mode 100644 index 000000000..b16244ce6 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtraces.yaml @@ -0,0 +1,283 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshtraces.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshTrace + listKind: MeshTraceList + plural: meshtraces + singular: meshtrace + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshTrace resource. + properties: + default: + description: MeshTrace configuration. + properties: + backends: + description: |- + A one element array of backend definition. + Envoy allows configuring only 1 backend, so the natural way of + representing that would be just one object. Unfortunately due to the + reasons explained in MADR 009-tracing-policy this has to be a one element + array for now. + items: + description: Only one of zipkin, datadog or openTelemetry can + be used. + properties: + datadog: + description: Datadog backend configuration. + properties: + splitService: + default: false + description: |- + Determines if datadog service name should be split based on traffic + direction and destination. For example, with `splitService: true` and a + `backend` service that communicates with a couple of databases, you would + get service names like `backend_INBOUND`, `backend_OUTBOUND_db1`, and + `backend_OUTBOUND_db2` in Datadog. + type: boolean + url: + description: |- + Address of Datadog collector, only host and port are allowed (no paths, + fragments etc.) + type: string + required: + - url + type: object + openTelemetry: + description: OpenTelemetry backend configuration. + properties: + endpoint: + description: Address of OpenTelemetry collector. + example: otel-collector:4317 + minLength: 1 + type: string + required: + - endpoint + type: object + type: + enum: + - Zipkin + - Datadog + - OpenTelemetry + type: string + zipkin: + description: Zipkin backend configuration. + properties: + apiVersion: + default: httpJson + description: |- + Version of the API. + https://github.com/envoyproxy/envoy/blob/v1.22.0/api/envoy/config/trace/v3/zipkin.proto#L66 + enum: + - httpJson + - httpProto + type: string + sharedSpanContext: + default: true + description: |- + Determines whether client and server spans will share the same span + context. + https://github.com/envoyproxy/envoy/blob/v1.22.0/api/envoy/config/trace/v3/zipkin.proto#L63 + type: boolean + traceId128bit: + default: false + description: Generate 128bit traces. + type: boolean + url: + description: Address of Zipkin collector. + type: string + required: + - url + type: object + required: + - type + type: object + maxItems: 1 + type: array + sampling: + description: |- + Sampling configuration. + Sampling is the process by which a decision is made on whether to + process/export a span or not. + properties: + client: + anyOf: + - type: integer + - type: string + default: 100 + description: |- + Target percentage of requests that will be force traced if the + 'x-client-trace-id' header is set. Mirror of client_sampling in Envoy + https://github.com/envoyproxy/envoy/blob/v1.22.0/api/envoy/config/filter/network/http_connection_manager/v2/http_connection_manager.proto#L127-L133 + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + overall: + anyOf: + - type: integer + - type: string + default: 100 + description: |- + Target percentage of requests will be traced + after all other sampling checks have been applied (client, force tracing, + random sampling). This field functions as an upper limit on the total + configured sampling rate. For instance, setting client to 100 + but overall to 1 will result in only 1% of client requests with + the appropriate headers to be force traced. Mirror of + overall_sampling in Envoy + https://github.com/envoyproxy/envoy/blob/v1.22.0/api/envoy/config/filter/network/http_connection_manager/v2/http_connection_manager.proto#L142-L150 + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + random: + anyOf: + - type: integer + - type: string + default: 100 + description: |- + Target percentage of requests that will be randomly selected for trace + generation, if not requested by the client or not forced. + Mirror of random_sampling in Envoy + https://github.com/envoyproxy/envoy/blob/v1.22.0/api/envoy/config/filter/network/http_connection_manager/v2/http_connection_manager.proto#L135-L140 + Either int or decimal represented as string. + x-kubernetes-int-or-string: true + type: object + tags: + description: |- + Custom tags configuration. You can add custom tags to traces based on + headers or literal values. + items: + description: |- + Custom tags configuration. + Only one of literal or header can be used. + properties: + header: + description: Tag taken from a header. + properties: + default: + description: |- + Default value to use if header is missing. + If the default is missing and there is no value the tag will not be + included. + type: string + name: + description: Name of the header. + type: string + required: + - name + type: object + literal: + description: Tag taken from literal value. + type: string + name: + description: Name of the tag. + type: string + required: + - name + type: object + type: array + type: object + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtrafficpermissions.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtrafficpermissions.yaml new file mode 100644 index 000000000..3e38acc06 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_meshtrafficpermissions.yaml @@ -0,0 +1,203 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: meshtrafficpermissions.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: MeshTrafficPermission + listKind: MeshTrafficPermissionList + plural: meshtrafficpermissions + singular: meshtrafficpermission + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .spec.targetRef.kind + name: TargetRef Kind + type: string + - jsonPath: .spec.targetRef.name + name: TargetRef Name + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 is the specification of the Kuma MeshTrafficPermission + resource. + properties: + from: + description: From list makes a match between clients and corresponding + configurations + items: + properties: + default: + description: |- + Default is a configuration specific to the group of clients referenced in + 'targetRef' + properties: + action: + description: 'Action defines a behavior for the specified + group of clients:' + enum: + - Allow + - Deny + - AllowWithShadowDeny + type: string + type: object + targetRef: + description: |- + TargetRef is a reference to the resource that represents a group of + clients. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify + cross mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + required: + - targetRef + type: object + type: array + targetRef: + description: |- + TargetRef is a reference to the resource the policy takes an effect on. + The resource could be either a real store object or virtual resource + defined inplace. + properties: + kind: + description: Kind of the referenced resource + enum: + - Mesh + - MeshSubset + - MeshGateway + - MeshService + - MeshExternalService + - MeshMultiZoneService + - MeshServiceSubset + - MeshHTTPRoute + type: string + labels: + additionalProperties: + type: string + description: |- + Labels are used to select group of MeshServices that match labels. Either Labels or + Name and Namespace can be used. + type: object + mesh: + description: Mesh is reserved for future use to identify cross + mesh resources. + type: string + name: + description: |- + Name of the referenced resource. Can only be used with kinds: `MeshService`, + `MeshServiceSubset` and `MeshGatewayRoute` + type: string + namespace: + description: |- + Namespace specifies the namespace of target resource. If empty only resources in policy namespace + will be targeted. + type: string + proxyTypes: + description: |- + ProxyTypes specifies the data plane types that are subject to the policy. When not specified, + all data plane types are targeted by the policy. + items: + enum: + - Sidecar + - Gateway + type: string + minItems: 1 + type: array + sectionName: + description: |- + SectionName is used to target specific section of resource. + For example, you can target port from MeshService.ports[] by its name. Only traffic to this port will be affected. + type: string + tags: + additionalProperties: + type: string + description: |- + Tags used to select a subset of proxies by tags. Can only be used with kinds + `MeshSubset` and `MeshServiceSubset` + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_proxytemplates.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_proxytemplates.yaml new file mode 100644 index 000000000..78b1d55e4 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_proxytemplates.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: proxytemplates.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ProxyTemplate + listKind: ProxyTemplateList + plural: proxytemplates + singular: proxytemplate + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ProxyTemplate resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_ratelimits.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_ratelimits.yaml new file mode 100644 index 000000000..85f1876eb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_ratelimits.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: ratelimits.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: RateLimit + listKind: RateLimitList + plural: ratelimits + singular: ratelimit + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma RateLimit resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_retries.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_retries.yaml new file mode 100644 index 000000000..10a4843e1 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_retries.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: retries.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: Retry + listKind: RetryList + plural: retries + singular: retry + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma Retry resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_serviceinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_serviceinsights.yaml new file mode 100644 index 000000000..827ea521d --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_serviceinsights.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: serviceinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ServiceInsight + listKind: ServiceInsightList + plural: serviceinsights + singular: serviceinsight + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ServiceInsight resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_timeouts.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_timeouts.yaml new file mode 100644 index 000000000..ba78d88c5 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_timeouts.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: timeouts.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: Timeout + listKind: TimeoutList + plural: timeouts + singular: timeout + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma Timeout resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficlogs.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficlogs.yaml new file mode 100644 index 000000000..ece8562e5 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficlogs.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: trafficlogs.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: TrafficLog + listKind: TrafficLogList + plural: trafficlogs + singular: trafficlog + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma TrafficLog resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficpermissions.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficpermissions.yaml new file mode 100644 index 000000000..9c79605af --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficpermissions.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: trafficpermissions.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: TrafficPermission + listKind: TrafficPermissionList + plural: trafficpermissions + singular: trafficpermission + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma TrafficPermission resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficroutes.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficroutes.yaml new file mode 100644 index 000000000..5bdd3ac85 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_trafficroutes.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: trafficroutes.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: TrafficRoute + listKind: TrafficRouteList + plural: trafficroutes + singular: trafficroute + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma TrafficRoute resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_traffictraces.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_traffictraces.yaml new file mode 100644 index 000000000..c224ea526 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_traffictraces.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: traffictraces.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: TrafficTrace + listKind: TrafficTraceList + plural: traffictraces + singular: traffictrace + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma TrafficTrace resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_virtualoutbounds.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_virtualoutbounds.yaml new file mode 100644 index 000000000..c4372dd0b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_virtualoutbounds.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: virtualoutbounds.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: VirtualOutbound + listKind: VirtualOutboundList + plural: virtualoutbounds + singular: virtualoutbound + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma VirtualOutbound resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegresses.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegresses.yaml new file mode 100644 index 000000000..143aaafdb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegresses.yaml @@ -0,0 +1,56 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zoneegresses.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ZoneEgress + listKind: ZoneEgressList + plural: zoneegresses + singular: zoneegress + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Zone name + jsonPath: .spec.zone + name: zone + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ZoneEgress resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegressinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegressinsights.yaml new file mode 100644 index 000000000..05746b39a --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneegressinsights.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zoneegressinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ZoneEgressInsight + listKind: ZoneEgressInsightList + plural: zoneegressinsights + singular: zoneegressinsight + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ZoneEgressInsight resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingresses.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingresses.yaml new file mode 100644 index 000000000..d02c5b35b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingresses.yaml @@ -0,0 +1,56 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zoneingresses.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ZoneIngress + listKind: ZoneIngressList + plural: zoneingresses + singular: zoneingress + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: Zone name + jsonPath: .spec.zone + name: zone + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ZoneIngress resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true + subresources: {} diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingressinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingressinsights.yaml new file mode 100644 index 000000000..ded86e6c2 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneingressinsights.yaml @@ -0,0 +1,51 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zoneingressinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ZoneIngressInsight + listKind: ZoneIngressInsightList + plural: zoneingressinsights + singular: zoneingressinsight + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ZoneIngressInsight + resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneinsights.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneinsights.yaml new file mode 100644 index 000000000..aad82d4be --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zoneinsights.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zoneinsights.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: ZoneInsight + listKind: ZoneInsightList + plural: zoneinsights + singular: zoneinsight + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma ZoneInsight resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/crds/kuma.io_zones.yaml b/charts/kuma/kuma/2.9.1/crds/kuma.io_zones.yaml new file mode 100644 index 000000000..12022fce9 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/crds/kuma.io_zones.yaml @@ -0,0 +1,50 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.16.3 + name: zones.kuma.io +spec: + group: kuma.io + names: + categories: + - kuma + kind: Zone + listKind: ZoneList + plural: zones + singular: zone + scope: Cluster + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + 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 + mesh: + description: |- + Mesh is the name of the Kuma mesh this resource belongs to. + It may be omitted for cluster-scoped resources. + type: string + metadata: + type: object + spec: + description: Spec is the specification of the Kuma Zone resource. + x-kubernetes-preserve-unknown-fields: true + type: object + served: true + storage: true diff --git a/charts/kuma/kuma/2.9.1/templates/NOTES.txt b/charts/kuma/kuma/2.9.1/templates/NOTES.txt new file mode 100644 index 000000000..228ac26e7 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/NOTES.txt @@ -0,0 +1,42 @@ +{{ .Chart.Name }} has been installed! + +Your release is named '{{ .Release.Name }}'. + +You can access the control-plane via either the GUI, kubectl, the HTTP API, or the kumactl CLI. +{{- if .Values.noHelmHooks }} + +------------------------------------------------------------------------------- + + WARNING + + When the "noHelmHooks" value is provided, you will need to manually delete + the "ValidatingWebhookConfiguration" responsible for validating {{ include "kuma.name" . }} resources + before you can uninstall Helm release. This is because the validation provided + by the webhook is not necessary during the release removal and might potentially + even prevent you from doing it. You can do this by running the following command: + + kubectl delete ValidatingWebhookConfiguration {{ include "kuma.name" . }}-validating-webhook-configuration + + WARNING + + When the "noHelmHooks" value is set, Helm will not automatically update + the CustomResourceDefinitions (CRDs) when upgrading release. You must manually + update the CRDs if the new {{ include "kuma.name" . }} version has changes + to the CRDs. You can achieve this by calling the following command: + + kumactl install crds --no-config | kubectl apply -f + +{{- if and .Values.experimental.ebpf.enabled (not .Values.cni.enabled) }} + + WARNING + + When the "noHelmHooks" value is set, Helm will not automatically uninstall + the eBPF resources. You will need to manually uninstall these resources after + uninstalling Helm release. To do this, run the following command: + + kumactl uninstall ebpf --cleanup-image-registry {{ .Values.global.image.registry }} --cleanup-image-repository {{ .Values.dataPlane.initImage.repository }} + +{{- end }} + +------------------------------------------------------------------------------- +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/_helpers.tpl b/charts/kuma/kuma/2.9.1/templates/_helpers.tpl new file mode 100644 index 000000000..a33fa04dc --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/_helpers.tpl @@ -0,0 +1,432 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "kuma.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +This is the Kuma version the chart is intended to be used with. +*/}} +{{- define "kuma.appVersion" -}} +{{- .Chart.AppVersion -}} +{{- end }} + +{{/* +This is only used in the `kuma.formatImage` function below. +*/}} +{{- define "kuma.defaultRegistry" -}} +docker.io/kumahq +{{- end }} + +{{- define "kuma.product" -}} +Kuma +{{- end }} + +{{- define "kuma.tagPrefix" -}} +{{- 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 "kuma.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 "kuma.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{- define "kuma.controlPlane.serviceName" -}} +{{- $defaultSvcName := printf "%s-control-plane" (include "kuma.name" .) -}} +{{ printf "%s" (default $defaultSvcName .Values.controlPlane.service.name) }} +{{- end }} + +{{- define "kuma.controlPlane.globalZoneSync.serviceName" -}} +{{- $defaultSvcName := printf "%s-global-zone-sync" (include "kuma.name" .) -}} +{{ printf "%s" (default $defaultSvcName .Values.controlPlane.globalZoneSyncService.name) }} +{{- end }} + +{{- define "kuma.ingress.serviceName" -}} +{{- $defaultSvcName := printf "%s-ingress" (include "kuma.name" .) -}} +{{ printf "%s" (default $defaultSvcName .Values.ingress.service.name) }} +{{- end }} + +{{- define "kuma.egress.serviceName" -}} +{{- $defaultSvcName := printf "%s-egress" (include "kuma.name" .) -}} +{{ printf "%s" (default $defaultSvcName .Values.egress.service.name) }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "kuma.labels" -}} +helm.sh/chart: {{ include "kuma.chart" . }} +{{ include "kuma.selectorLabels" . }} +{{- if (include "kuma.appVersion" .) }} +app.kubernetes.io/version: {{ (include "kuma.appVersion" .) | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "kuma.selectorLabels" -}} +app.kubernetes.io/name: {{ include "kuma.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +CNI labels +*/}} +{{- define "kuma.cniLabels" -}} +app: {{ include "kuma.name" . }}-cni +{{ include "kuma.labels" . }} +{{- end }} + +{{/* +control plane labels +*/}} +{{- define "kuma.cpLabels" -}} +app: {{ include "kuma.name" . }}-control-plane +{{- range $key, $value := $.Values.controlPlane.extraLabels }} +{{ $key | quote }}: {{ $value | quote }} +{{- end }} +{{ include "kuma.labels" . }} +{{- end }} + +{{/* +control plane deployment annotations +*/}} +{{- define "kuma.cpDeploymentAnnotations" -}} +{{- range $key, $value := $.Values.controlPlane.deploymentAnnotations }} +{{ $key | quote }}: {{ $value | quote }} +{{- end }} +{{- end }} + +{{/* +ingress labels +*/}} +{{- define "kuma.ingressLabels" -}} +app: {{ include "kuma.name" . }}-ingress +{{- range $key, $value := .Values.ingress.extraLabels }} +{{ $key | quote }}: {{ $value | quote }} +{{- end }} +{{ include "kuma.labels" . }} +{{- end }} + +{{/* +egress labels +*/}} +{{- define "kuma.egressLabels" -}} +app: {{ include "kuma.name" . }}-egress +{{ range $key, $value := .Values.egress.extraLabels }} +{{ $key | quote }}: {{ $value | quote }} +{{ end }} +{{- include "kuma.labels" . }} +{{- end }} + +{{/* +CNI selector labels +*/}} +{{- define "kuma.cniSelectorLabels" -}} +app: {{ include "kuma.name" . }}-cni +{{ include "kuma.selectorLabels" . }} +{{- end }} + +{{/* +params: { dns: { policy?, config: {nameservers?, searches?}} } +returns: formatted dnsConfig +*/}} +{{- define "kuma.dnsConfig" -}} +{{- $dns := .dns }} +{{- if $dns.policy }} +dnsPolicy: {{ $dns.policy }} +{{- end }} +{{- if or (gt (len $dns.config.nameservers) 0) (gt (len $dns.config.searches) 0) }} +dnsConfig: + {{- if gt (len $dns.config.nameservers) 0 }} + nameservers: + {{- range $nameserver := $dns.config.nameservers }} + - {{ $nameserver }} + {{- end }} + {{- end }} + {{- if gt (len $dns.config.searches) 0 }} + searches: + {{- range $search := $dns.config.searches }} + - {{ $search }} + {{- end }} + {{- end }} +{{- end }} +{{- end -}} + +{{/* +params: { image: { registry?, repository, tag? }, root: $ } +returns: formatted image string +*/}} +{{- define "kuma.formatImage" -}} +{{- $img := .image }} +{{- $root := .root }} +{{- $registry := ($img.registry | default $root.Values.global.image.registry) -}} +{{- $repo := ($img.repository | required "Must specify image repository") -}} +{{- $product := (include "kuma.product" .) }} +{{- $tagPrefix := (include "kuma.tagPrefix" .) }} +{{- $expectedVersion := (include "kuma.appVersion" $root) }} +{{- if + and + $root.Values.global.image.tag + (ne $root.Values.global.image.tag (include "kuma.appVersion" $root)) + (eq $root.Values.global.image.registry (include "kuma.defaultRegistry" .)) +-}} +{{- fail ( + printf "This chart only supports %s version %q but %sglobal.image.tag is set to %q. Set %sglobal.image.tag to %q or skip this check by setting %s*.image.tag for each individual component." + $product $expectedVersion $tagPrefix $root.Values.global.image.tag $tagPrefix $expectedVersion $tagPrefix +) -}} +{{- end -}} +{{- $defaultTag := ($root.Values.global.image.tag | default (include "kuma.appVersion" $root)) -}} +{{- $tag := ($img.tag | default $defaultTag) -}} +{{- printf "%s/%s:%s" $registry $repo $tag -}} +{{- end -}} + +{{- define "kuma.parentEnv" -}} +{{- end -}} + +{{- define "kuma.parentSecrets" -}} +{{- end -}} + +{{- define "kuma.pluginPoliciesEnabled" -}} +{{- $list := list -}} +{{- range $k, $v := .Values.plugins.policies -}} +{{- if $v -}} +{{- $list = append $list (printf "%s" $k) -}} +{{- end -}} +{{- end -}} +{{ join "," $list }} +{{- end -}} + +{{- define "kuma.transparentProxyConfigMapName" -}} +{{- if .Values.transparentProxy.configMap.name }} +{{- .Values.transparentProxy.configMap.name | trunc 253 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-transparent-proxy-config" .Chart.Name }} +{{- end }} +{{- end }} + +{{- define "kuma.defaultEnv" -}} +env: +{{ include "kuma.parentEnv" . }} +- name: KUMA_ENVIRONMENT + value: "kubernetes" +- name: KUMA_STORE_TYPE + value: "kubernetes" +- name: KUMA_STORE_KUBERNETES_SYSTEM_NAMESPACE + value: {{ .Release.Namespace | quote }} +- name: KUMA_RUNTIME_KUBERNETES_CONTROL_PLANE_SERVICE_NAME + value: {{ include "kuma.controlPlane.serviceName" . }} +- name: KUMA_GENERAL_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/tls-cert/tls.crt +- name: KUMA_GENERAL_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/tls-cert/tls.key +{{- if eq .Values.controlPlane.mode "zone" }} +- name: KUMA_MULTIZONE_ZONE_GLOBAL_ADDRESS + value: {{ .Values.controlPlane.kdsGlobalAddress }} +{{- end }} +- name: KUMA_DP_SERVER_HDS_ENABLED + value: "false" +- name: KUMA_API_SERVER_READ_ONLY + value: "true" +- name: KUMA_RUNTIME_KUBERNETES_ADMISSION_SERVER_PORT + value: {{ .Values.controlPlane.admissionServerPort | default "5443" | quote }} +- name: KUMA_RUNTIME_KUBERNETES_ADMISSION_SERVER_CERT_DIR + value: /var/run/secrets/kuma.io/tls-cert +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_CNI_ENABLED + value: {{ .Values.cni.enabled | quote }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_SIDECAR_CONTAINER_IMAGE + value: {{ include "kuma.formatImage" (dict "image" .Values.dataPlane.image "root" $) | quote }} +- name: KUMA_INJECTOR_INIT_CONTAINER_IMAGE + value: {{ include "kuma.formatImage" (dict "image" .Values.dataPlane.initImage "root" $) | quote }} +{{- if .Values.dataPlane.dnsLogging }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_BUILTIN_DNS_LOGGING + value: "true" +{{- end }} +{{- if and .Values.transparentProxy.configMap.enabled .Values.transparentProxy.configMap.config }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_TRANSPARENT_PROXY_CONFIGMAP_NAME + value: {{ include "kuma.transparentProxyConfigMapName" . | quote }} +{{- end }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_CA_CERT_FILE + value: /var/run/secrets/kuma.io/tls-cert/ca.crt +- name: KUMA_DEFAULTS_SKIP_MESH_CREATION + value: {{ .Values.controlPlane.defaults.skipMeshCreation | quote }} +- name: KUMA_MODE + value: {{ .Values.controlPlane.mode | quote }} +{{- if .Values.controlPlane.zone }} +- name: KUMA_MULTIZONE_ZONE_NAME + value: {{ .Values.controlPlane.zone | quote }} +{{- end }} +{{- if .Values.controlPlane.tls.apiServer.secretName }} +- name: KUMA_API_SERVER_HTTPS_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/api-server-tls-cert/tls.crt +- name: KUMA_API_SERVER_HTTPS_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/api-server-tls-cert/tls.key +{{- end }} +{{- if .Values.controlPlane.tls.apiServer.clientCertsSecretName }} +- name: KUMA_API_SERVER_AUTH_CLIENT_CERTS_DIR + value: /var/run/secrets/kuma.io/api-server-client-certs/ +{{- end }} +{{- if and (eq .Values.controlPlane.mode "global") (or .Values.controlPlane.tls.kdsGlobalServer.secretName .Values.controlPlane.tls.kdsGlobalServer.create) }} +- name: KUMA_MULTIZONE_GLOBAL_KDS_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/kds-server-tls-cert/tls.crt +- name: KUMA_MULTIZONE_GLOBAL_KDS_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/kds-server-tls-cert/tls.key +{{- end }} +{{- if and (eq .Values.controlPlane.mode "zone") (or .Values.controlPlane.tls.kdsZoneClient.secretName .Values.controlPlane.tls.kdsZoneClient.create) }} +- name: KUMA_MULTIZONE_ZONE_KDS_ROOT_CA_FILE + value: /var/run/secrets/kuma.io/kds-client-tls-cert/ca.crt +{{- end }} +- name: KUMA_API_SERVER_AUTHN_LOCALHOST_IS_ADMIN + value: "false" +- name: KUMA_RUNTIME_KUBERNETES_ALLOWED_USERS + value: "system:serviceaccount:{{ .Release.Namespace }}:{{ include "kuma.name" . }}-control-plane" +{{- if .Values.experimental.sidecarContainers }} +- name: KUMA_EXPERIMENTAL_SIDECAR_CONTAINERS + value: "true" +{{- end }} +{{- if .Values.cni.enabled }} +- name: KUMA_RUNTIME_KUBERNETES_NODE_TAINT_CONTROLLER_ENABLED + value: "true" +- name: KUMA_RUNTIME_KUBERNETES_NODE_TAINT_CONTROLLER_CNI_APP + value: "{{ include "kuma.name" . }}-cni" +- name: KUMA_RUNTIME_KUBERNETES_NODE_TAINT_CONTROLLER_CNI_NAMESPACE + value: {{ .Values.cni.namespace }} +{{- end }} +{{- if .Values.experimental.ebpf.enabled }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_ENABLED + value: "true" +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_INSTANCE_IP_ENV_VAR_NAME + value: {{ .Values.experimental.ebpf.instanceIPEnvVarName }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_BPFFS_PATH + value: {{ .Values.experimental.ebpf.bpffsPath }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_CGROUP_PATH + value: {{ .Values.experimental.ebpf.cgroupPath }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_TC_ATTACH_IFACE + value: {{ .Values.experimental.ebpf.tcAttachIface }} +- name: KUMA_RUNTIME_KUBERNETES_INJECTOR_EBPF_PROGRAMS_SOURCE_PATH + value: {{ .Values.experimental.ebpf.programsSourcePath }} +{{- end }} +{{- if .Values.controlPlane.tls.kdsZoneClient.skipVerify }} +- name: KUMA_MULTIZONE_ZONE_KDS_TLS_SKIP_VERIFY + value: "true" +{{- end }} +- name: KUMA_PLUGIN_POLICIES_ENABLED + value: {{ include "kuma.pluginPoliciesEnabled" . | quote }} +{{- if .Values.controlPlane.supportGatewaySecretsInAllNamespaces }} +- name: KUMA_RUNTIME_KUBERNETES_SUPPORT_GATEWAY_SECRETS_IN_ALL_NAMESPACES + value: true +{{- end }} +{{- end }} + +{{- define "kuma.controlPlane.tls.general.caSecretName" -}} +{{ .Values.controlPlane.tls.general.caSecretName | default .Values.controlPlane.tls.general.secretName | default (printf "%s-tls-cert" (include "kuma.name" .)) | quote }} +{{- end }} + +{{- define "kuma.universal.defaultEnv" -}} +{{ if eq .Values.controlPlane.mode "zone" }} + {{ if .Values.ingress.enabled }} + {{ fail "Can't have ingress.enabled when running controlPlane.mode=='universal'" }} + {{ end }} + {{ if .Values.egress.enabled }} + {{ fail "Can't have egress.enabled when running controlPlane.mode=='universal'" }} + {{ end }} +{{ end }} + +env: +- name: KUMA_PLUGIN_POLICIES_ENABLED + value: {{ include "kuma.pluginPoliciesEnabled" . | quote }} +- name: KUMA_GENERAL_WORK_DIR + value: "/tmp/kuma" +- name: KUMA_ENVIRONMENT + value: "universal" +- name: KUMA_STORE_TYPE + value: "postgres" +- name: KUMA_STORE_POSTGRES_PORT + value: "{{ .Values.postgres.port }}" +- name: KUMA_DEFAULTS_SKIP_MESH_CREATION + value: {{ .Values.controlPlane.defaults.skipMeshCreation | quote }} +{{ if and (eq .Values.controlPlane.mode "zone") .Values.controlPlane.tls.general.secretName }} +- name: KUMA_GENERAL_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/tls-cert/tls.crt +- name: KUMA_GENERAL_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/tls-cert/tls.key +{{ end }} +- name: KUMA_MODE + value: {{ .Values.controlPlane.mode | quote }} +{{- if eq .Values.controlPlane.mode "zone" }} +- name: KUMA_MULTIZONE_ZONE_GLOBAL_ADDRESS + value: {{ .Values.controlPlane.kdsGlobalAddress }} +{{- end }} +{{- if .Values.controlPlane.zone }} +- name: KUMA_MULTIZONE_ZONE_NAME + value: {{ .Values.controlPlane.zone | quote }} +{{- end }} +{{- if and (eq .Values.controlPlane.mode "zone") (or .Values.controlPlane.tls.kdsZoneClient.secretName .Values.controlPlane.tls.kdsZoneClient.create) }} +- name: KUMA_MULTIZONE_ZONE_KDS_ROOT_CA_FILE + value: /var/run/secrets/kuma.io/kds-client-tls-cert/ca.crt +{{- end }} +{{- if .Values.controlPlane.tls.kdsZoneClient.skipVerify }} +- name: KUMA_MULTIZONE_ZONE_KDS_TLS_SKIP_VERIFY + value: "true" +{{- end }} +{{- if .Values.controlPlane.tls.apiServer.secretName }} +- name: KUMA_API_SERVER_HTTPS_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/api-server-tls-cert/tls.crt +- name: KUMA_API_SERVER_HTTPS_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/api-server-tls-cert/tls.key +{{- end }} +{{- if .Values.controlPlane.tls.apiServer.clientCertsSecretName }} +- name: KUMA_API_SERVER_AUTH_CLIENT_CERTS_DIR + value: /var/run/secrets/kuma.io/api-server-client-certs/ +{{- end }} +{{- if .Values.controlPlane.tls.kdsGlobalServer.secretName }} +- name: KUMA_MULTIZONE_GLOBAL_KDS_TLS_CERT_FILE + value: /var/run/secrets/kuma.io/kds-server-tls-cert/tls.crt +- name: KUMA_MULTIZONE_GLOBAL_KDS_TLS_KEY_FILE + value: /var/run/secrets/kuma.io/kds-server-tls-cert/tls.key +{{- end }} +- name: KUMA_STORE_POSTGRES_TLS_MODE + value: {{ .Values.postgres.tls.mode }} +{{- if or (eq .Values.postgres.tls.mode "verifyCa") (eq .Values.postgres.tls.mode "verifyFull") }} +{{- if empty .Values.postgres.tls.caSecretName }} +{{ fail "if mode is 'verifyCa' or 'verifyFull' then you must provide .Values.postgres.tls.caSecretName" }} +{{- end }} +{{- if .Values.postgres.tls.secretName }} +- name: KUMA_STORE_POSTGRES_TLS_CERT_PATH + value: /var/run/secrets/kuma.io/postgres-tls-cert/tls.crt +- name: KUMA_STORE_POSTGRES_TLS_KEY_PATH + value: /var/run/secrets/kuma.io/postgres-tls-cert/tls.key +{{- end }} +{{- if .Values.postgres.tls.caSecretName }} +- name: KUMA_STORE_POSTGRES_TLS_CA_PATH + value: /var/run/secrets/kuma.io/postgres-tls-cert/ca.crt +{{- end }} +{{- if .Values.postgres.tls.disableSSLSNI }} +- name: KUMA_STORE_POSTGRES_TLS_DISABLE_SSLSNI + value: {{ .Values.postgres.tls.disableSSLSNI }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cni-configmap.yaml b/charts/kuma/kuma/2.9.1/templates/cni-configmap.yaml new file mode 100644 index 000000000..8d27de9ef --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cni-configmap.yaml @@ -0,0 +1,22 @@ +{{- if and .Values.cni.enabled (not .Values.experimental.ebpf.enabled) }} +kind: ConfigMap +apiVersion: v1 +metadata: + name: {{ include "kuma.name" . }}-cni-config + namespace: {{ .Values.cni.namespace }} + labels: {{ include "kuma.cniLabels" . | nindent 4 }} +data: + # The CNI network configuration to add to the plugin chain on each node. + cni_network_config: |- + { + "cniVersion": "0.3.1", + "name": "kuma-cni", + "type": "kuma-cni", + "log_level": "{{ .Values.cni.logLevel }}", + "kubernetes": { + "kubeconfig": "__KUBECONFIG_FILEPATH__", + "cni_bin_dir": "{{ .Values.cni.binDir }}", + "exclude_namespaces": [ "kube-system" ] + } + } + {{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cni-daemonset.yaml b/charts/kuma/kuma/2.9.1/templates/cni-daemonset.yaml new file mode 100644 index 000000000..b5d8db761 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cni-daemonset.yaml @@ -0,0 +1,152 @@ +{{- if .Values.cni.enabled }} +kind: DaemonSet +apiVersion: apps/v1 +metadata: + name: {{ include "kuma.name" . }}-cni-node + namespace: {{ .Values.cni.namespace }} + annotations: + ignore-check.kube-linter.io/run-as-non-root: "The container installs a CNI plugin" + labels: {{- include "kuma.cniLabels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "kuma.cniSelectorLabels" . | nindent 6 }} + updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + template: + metadata: + labels: + {{- include "kuma.cniSelectorLabels" . | nindent 8 }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/cni-configmap.yaml") . | sha256sum }} + {{- range $key, $value := .Values.cni.podAnnotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + spec: + # This, along with the CriticalAddonsOnly toleration below, + # marks the pod as a critical add-on, ensuring it gets + # priority scheduling and that its resources are reserved + # if it ever gets evicted. + priorityClassName: system-node-critical + {{- with .Values.cni.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.cni.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + tolerations: + # Make sure kuma-cni-node gets scheduled on all nodes. + - effect: NoSchedule + operator: Exists + # Mark the pod as a critical add-on for rescheduling. + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + serviceAccountName: {{ include "kuma.name" . }}-cni + # Minimize downtime during a rolling upgrade or deletion; tell Kubernetes to do a "force + # deletion": https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods. + terminationGracePeriodSeconds: 5 + securityContext: + {{- toYaml .Values.cni.podSecurityContext | trim | nindent 8 }} + containers: + - name: install-cni + imagePullPolicy: {{ .Values.cni.image.imagePullPolicy }} + {{- if not .Values.experimental.ebpf.enabled }} + image: {{ include "kuma.formatImage" (dict "image" .Values.cni.image "root" $) | quote }} + readinessProbe: + initialDelaySeconds: {{ .Values.cni.delayStartupSeconds }} + exec: + command: + - cat + - /tmp/ready + command: [ "sh", "-c", "--" ] + args: [ "sleep {{.Values.cni.delayStartupSeconds}} && exec /install-cni" ] + {{- else }} + {{- with .Values.cni.experimental.imageEbpf }} + image: {{ printf "%s/%s:%s" .registry .repository .tag | quote }} + {{- end }} + args: + - /app/mbctl + - --mode=kuma + - --use-reconnect=true + - --cni-mode=true + {{- if eq .Values.cni.logLevel "debug" }} + - --debug=true + {{- end }} + lifecycle: + preStop: + exec: + command: + - make + - --keep-going + - clean + {{- end }} + securityContext: + {{- toYaml .Values.cni.containerSecurityContext | trim | nindent 12 }} + {{- if .Values.experimental.ebpf.enabled }} + privileged: true + {{- end }} + {{- if not .Values.experimental.ebpf.enabled }} + env: + # Name of the CNI config file to create. + - name: CNI_CONF_NAME + value: "{{ .Values.cni.confName }}" + # The CNI network config to install on each node. + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + name: {{ include "kuma.name" . }}-cni-config + key: cni_network_config + - name: CNI_NET_DIR + value: "{{ .Values.cni.netDir }}" + # If true, deploy as a chained CNI plugin, otherwise deploy as a standalone CNI + - name: CHAINED_CNI_PLUGIN + value: "{{ .Values.cni.chained }}" + - name: CNI_LOG_LEVEL + value: "{{ .Values.cni.logLevel }}" + {{- end }} + resources: + {{- toYaml .Values.cni.resources | trim | nindent 12 }} + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + {{- if .Values.experimental.ebpf.enabled }} + - mountPath: /sys/fs/cgroup + name: sys-fs-cgroup + - mountPath: /host/proc + name: host-proc + - mountPath: /host/var/run + name: host-var-run + mountPropagation: Bidirectional + {{- end }} + - name: tmp + mountPath: /tmp + volumes: + # Used to install CNI. + - name: cni-bin-dir + hostPath: + path: {{ .Values.cni.binDir }} + - name: cni-net-dir + hostPath: + path: {{ .Values.cni.netDir }} + {{- if .Values.experimental.ebpf.enabled }} + - hostPath: + path: /var/run + name: host-var-run + - hostPath: + path: /sys/fs/cgroup + name: sys-fs-cgroup + - hostPath: + path: /proc + name: host-proc + {{- end }} + - name: tmp + emptyDir: {} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cni-rbac.yaml b/charts/kuma/kuma/2.9.1/templates/cni-rbac.yaml new file mode 100644 index 000000000..07af2b215 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cni-rbac.yaml @@ -0,0 +1,51 @@ +{{- if .Values.cni.enabled }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "kuma.name" . }}-cni + namespace: {{ .Values.cni.namespace }} + labels: {{ include "kuma.cniLabels" . | nindent 4 }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-cni + labels: + {{ include "kuma.cniLabels" . | nindent 4 }} +rules: + - apiGroups: [""] + resources: + - nodes + verbs: + - get + - apiGroups: [""] + resources: + - pods + verbs: + - get + {{- if .Values.experimental.ebpf.enabled }} + - list + - watch + {{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-cni + labels: + {{ include "kuma.cniLabels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-cni +subjects: + - kind: ServiceAccount + name: {{ include "kuma.name" . }}-cni + namespace: {{ .Values.cni.namespace }} + {{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-configmap.yaml b/charts/kuma/kuma/2.9.1/templates/cp-configmap.yaml new file mode 100644 index 000000000..b2c94ed4d --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-configmap.yaml @@ -0,0 +1,46 @@ +{{ $kumaCpLabels := include "kuma.cpLabels" . }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "kuma.name" . }}-control-plane-config + namespace: {{ .Release.Namespace }} + labels: {{ $kumaCpLabels | nindent 4 }} +data: + config.yaml: | + # use this file to override default configuration of `kuma-cp` + # + # see conf/kuma-cp.conf.yml for available settings + {{ if .Values.controlPlane.config }} + {{ .Values.controlPlane.config | nindent 4 }} + {{ end }} + +{{- $releaseNamespace := .Release.Namespace}} +{{- range $extraConfigMap := .Values.controlPlane.extraConfigMaps }} +{{- if $extraConfigMap.values }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ $extraConfigMap.name }} + namespace: {{ $releaseNamespace }} + labels: {{ $kumaCpLabels | nindent 4 }} +data: + {{- range $fileName, $fileContents := $extraConfigMap.values }} + {{- $fileName | nindent 2 }}: | + {{- $fileContents | nindent 4 }} + {{- end }} +{{- end }} +{{- end }} +{{- if and .Values.transparentProxy.configMap.enabled .Values.transparentProxy.configMap.config }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "kuma.transparentProxyConfigMapName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- $kumaCpLabels | nindent 4 }} +data: + config.yaml: | + {{- .Values.transparentProxy.configMap.config | toYaml | nindent 4 }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-deployment.yaml b/charts/kuma/kuma/2.9.1/templates/cp-deployment.yaml new file mode 100644 index 000000000..1111b149b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-deployment.yaml @@ -0,0 +1,412 @@ +{{ $kdsGlobalServerTLSSecretName := "" }} +{{ if eq .Values.controlPlane.mode "global" }} + {{ $kdsGlobalServerTLSSecretName = .Values.controlPlane.tls.kdsGlobalServer.secretName }} + {{ if and .Values.controlPlane.tls.kdsGlobalServer.create (not $kdsGlobalServerTLSSecretName) }} + {{ $kdsGlobalServerTLSSecretName = print (include "kuma.name" .) "-kds-global-server-tls" }} + {{ end }} +{{ end }} + +{{ $kdsZoneClientTLSSecretName := "" }} +{{ if eq .Values.controlPlane.mode "zone" }} + {{ $kdsZoneClientTLSSecretName = .Values.controlPlane.tls.kdsZoneClient.secretName }} + {{ if and .Values.controlPlane.tls.kdsZoneClient.create (not $kdsZoneClientTLSSecretName) }} + {{ $kdsZoneClientTLSSecretName = print (include "kuma.name" .) "-kds-zone-client-tls" }} + {{ end }} +{{ end }} + +{{ if not (or (eq .Values.controlPlane.mode "zone") (eq .Values.controlPlane.mode "global") (eq .Values.controlPlane.mode "standalone")) }} + {{ $msg := printf "controlPlane.mode invalid got:'%s' supported values: global,zone,standalone" .Values.controlPlane.mode }} + {{ fail $msg }} +{{ end }} +{{ if eq .Values.controlPlane.mode "zone" }} + {{ if not (empty .Values.controlPlane.zone) }} + {{ if gt (len .Values.controlPlane.zone) 253 }} + {{ fail "controlPlane.zone must be no more than 253 characters" }} + {{ else }} + {{ if not (regexMatch "^[a-z0-9]([-a-z0-9]*[a-z0-9])?(\\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*$" .Values.controlPlane.zone) }} + {{ fail "controlPlane.zone must consist of lower case alphanumeric characters, '-' or '.', and must start and end with an alphanumeric character" }} + {{ end }} + {{ end }} + {{ end }} + {{ if not (empty .Values.controlPlane.kdsGlobalAddress) }} + {{ $url := urlParse .Values.controlPlane.kdsGlobalAddress }} + {{ if not (or (eq $url.scheme "grpcs") (eq $url.scheme "grpc")) }} + {{ $msg := printf "controlPlane.kdsGlobalAddress must be a url with scheme grpcs:// or grpc:// got:'%s'" .Values.controlPlane.kdsGlobalAddress }} + {{ fail $msg }} + {{ end }} + {{ end }} +{{ else }} + {{ if not (empty .Values.controlPlane.zone) }} + {{ fail "Can't specify a controlPlane.zone when controlPlane.mode!='zone'" }} + {{ end }} + {{ if not (empty .Values.controlPlane.kdsGlobalAddress) }} + {{ fail "Can't specify a controlPlane.kdsGlobalAddress when controlPlane.mode!='zone'" }} + {{ end }} +{{ end }} + +{{- $defaultEnv := include "kuma.defaultEnv" . | fromYaml | pluck "env" | first }} +{{- if eq .Values.controlPlane.environment "universal" }} +{{- $defaultEnv = include "kuma.universal.defaultEnv" . | fromYaml | pluck "env" | first }} +{{- end }} +{{- $defaultEnvDict := dict }} +{{- range $index, $item := $defaultEnv }} +{{- $name := $item.name | upper }} +{{- $defaultEnvDict := set $defaultEnvDict $name $item.value }} +{{- end }} +{{- $envVarsCopy := deepCopy .Values.controlPlane.envVars }} +{{- $mergedEnv := merge $envVarsCopy $defaultEnvDict }} +{{- $defaultSecrets := include "kuma.parentSecrets" . | fromYaml }} +{{- $extraSecrets := .Values.controlPlane.extraSecrets }} +{{- $mergedSecrets := merge $extraSecrets $defaultSecrets }} + +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} + annotations: {{ include "kuma.cpDeploymentAnnotations" . | nindent 4 }} +spec: + {{- if not .Values.controlPlane.autoscaling.enabled }} + replicas: {{ .Values.controlPlane.replicas }} + {{- end }} + minReadySeconds: {{ .Values.controlPlane.minReadySeconds }} + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 0 + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-control-plane + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/cp-configmap.yaml") . | sha256sum }} + {{- if .Values.restartOnSecretChange }} + checksum/tls-secrets: {{ include (print $.Template.BasePath "/cp-webhooks-and-secrets.yaml") . | sha256sum }} + {{- end }} + {{- range $key, $value := $.Values.controlPlane.podAnnotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: {{ include "kuma.cpLabels" . | nindent 8 }} + spec: + {{- with .Values.controlPlane.affinity }} + affinity: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + {{- with .Values.controlPlane.topologySpreadConstraints }} + topologySpreadConstraints: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + securityContext: + {{- toYaml .Values.controlPlane.podSecurityContext | trim | nindent 8 }} + serviceAccountName: {{ include "kuma.name" . }}-control-plane + automountServiceAccountToken: {{ .Values.controlPlane.automountServiceAccountToken }} + {{- with .Values.controlPlane.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.controlPlane.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + hostNetwork: {{ .Values.controlPlane.hostNetwork }} + terminationGracePeriodSeconds: {{ .Values.controlPlane.terminationGracePeriodSeconds }} + {{ include "kuma.dnsConfig" (dict "dns" .Values.controlPlane.dns) | nindent 6 | trim }} + {{- if (eq .Values.controlPlane.environment "universal") }} + initContainers: + - name: migration + image: {{ include "kuma.formatImage" (dict "image" .Values.controlPlane.image "root" $) | quote }} + imagePullPolicy: {{ .Values.controlPlane.image.pullPolicy }} + securityContext: + {{- toYaml .Values.controlPlane.containerSecurityContext | trim | nindent 12 }} + env: + {{- range $key, $value := $mergedEnv }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $element := .Values.controlPlane.secrets }} + - name: {{ $element.Env }} + valueFrom: + secretKeyRef: + name: {{ $element.Secret }} + key: {{ $element.Key }} + {{- end }} + args: + - migrate + - up + - --log-level=info + - --config-file=/etc/kuma.io/kuma-control-plane/config.yaml + resources: + {{- if .Values.controlPlane.resources }} + {{- .Values.controlPlane.resources | toYaml | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.postgres.tls.caSecretName }} + - name: postgres-tls-cert-ca + subPath: ca.crt + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/ca.crt + readOnly: true + {{- end }} + {{- if .Values.postgres.tls.secretName }} + - name: postgres-tls-cert + subPath: tls.crt + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/tls.crt + readOnly: true + - name: postgres-tls-cert + subPath: tls.key + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/tls.key + readOnly: true + {{- end }} + - name: {{ include "kuma.name" . }}-control-plane-config + mountPath: /etc/kuma.io/kuma-control-plane + readOnly: true + {{- end }} + containers: + - name: control-plane + image: {{ include "kuma.formatImage" (dict "image" .Values.controlPlane.image "root" $) | quote }} + imagePullPolicy: {{ .Values.controlPlane.image.pullPolicy }} + securityContext: + {{- toYaml .Values.controlPlane.containerSecurityContext | trim | nindent 12 }} + env: + {{- if .Values.controlPlane.envVarEntries }} + {{- .Values.controlPlane.envVarEntries | toYaml | nindent 12 }} + {{- end }} + {{- range $key, $value := $mergedEnv }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $element := .Values.controlPlane.secrets }} + - name: {{ $element.Env }} + valueFrom: + secretKeyRef: + name: {{ $element.Secret }} + key: {{ $element.Key }} + {{- end }} + - name: KUMA_INTER_CP_CATALOG_INSTANCE_ADDRESS + valueFrom: + fieldRef: + fieldPath: status.podIP + - name: GOMEMLIMIT + valueFrom: + resourceFieldRef: + containerName: control-plane + resource: limits.memory + - name: GOMAXPROCS + valueFrom: + resourceFieldRef: + containerName: control-plane + resource: limits.cpu + args: + - run + - --log-level={{ .Values.controlPlane.logLevel }} + - --log-output-path={{ .Values.controlPlane.logOutputPath }} + - --config-file=/etc/kuma.io/kuma-control-plane/config.yaml + ports: + - containerPort: 5680 + name: diagnostics + protocol: TCP + - containerPort: 5681 + - containerPort: 5682 + - containerPort: {{ .Values.controlPlane.admissionServerPort | default "5443" }} + {{- if ne .Values.controlPlane.mode "global" }} + - containerPort: 5678 + {{- end }} + livenessProbe: + timeoutSeconds: 10 + httpGet: + path: /healthy + port: 5680 + readinessProbe: + timeoutSeconds: 10 + httpGet: + path: /ready + port: 5680 + resources: + {{- if .Values.controlPlane.resources }} + {{- .Values.controlPlane.resources | toYaml | nindent 12 }} + {{- end }} + {{ with .Values.controlPlane.lifecycle }} + lifecycle: {{ . | toYaml | nindent 14 }} + {{ end }} + volumeMounts: + {{- if eq .Values.controlPlane.environment "kubernetes" }} + {{- if not .Values.controlPlane.automountServiceAccountToken }} + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: serviceaccount-token + readOnly: true + {{- end }} + - name: general-tls-cert + mountPath: /var/run/secrets/kuma.io/tls-cert/tls.crt + subPath: tls.crt + readOnly: true + - name: general-tls-cert + mountPath: /var/run/secrets/kuma.io/tls-cert/tls.key + subPath: tls.key + readOnly: true + - name: general-tls-cert{{- if .Values.controlPlane.tls.general.caSecretName }}-ca{{- end }} + mountPath: /var/run/secrets/kuma.io/tls-cert/ca.crt + subPath: ca.crt + readOnly: true + {{- end }} + {{- if and (eq .Values.controlPlane.environment "universal") (eq .Values.controlPlane.mode "zone") }} + {{- if .Values.controlPlane.tls.general.secretName }} + - name: general-tls-cert + mountPath: /var/run/secrets/kuma.io/tls-cert/tls.crt + subPath: tls.crt + readOnly: true + - name: general-tls-cert + mountPath: /var/run/secrets/kuma.io/tls-cert/tls.key + subPath: tls.key + readOnly: true + - name: general-tls-cert{{- if .Values.controlPlane.tls.general.caSecretName }}-ca{{- end }} + mountPath: /var/run/secrets/kuma.io/tls-cert/ca.crt + subPath: ca.crt + readOnly: true + {{- end }} + {{- end }} + - name: {{ include "kuma.name" . }}-control-plane-config + mountPath: /etc/kuma.io/kuma-control-plane + readOnly: true + {{- if .Values.controlPlane.tls.apiServer.secretName }} + - name: api-server-tls-cert + mountPath: /var/run/secrets/kuma.io/api-server-tls-cert + readOnly: true + {{- end }} + {{- if .Values.postgres.tls.caSecretName }} + - name: postgres-tls-cert-ca + subPath: ca.crt + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/ca.crt + readOnly: true + {{- end }} + {{- if .Values.postgres.tls.secretName }} + - name: postgres-tls-cert + subPath: tls.crt + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/tls.crt + readOnly: true + - name: postgres-tls-cert + subPath: tls.key + mountPath: /var/run/secrets/kuma.io/postgres-tls-cert/tls.key + readOnly: true + {{- end }} + {{- if .Values.controlPlane.tls.apiServer.clientCertsSecretName }} + - name: api-server-client-certs + mountPath: /var/run/secrets/kuma.io/api-server-client-certs + readOnly: true + {{- end }} + {{- if $kdsGlobalServerTLSSecretName }} + - name: kds-server-tls-cert + mountPath: /var/run/secrets/kuma.io/kds-server-tls-cert + readOnly: true + {{- end }} + {{- if $kdsZoneClientTLSSecretName }} + - name: kds-client-tls-cert + mountPath: /var/run/secrets/kuma.io/kds-client-tls-cert + readOnly: true + {{- end }} + {{- range $extraConfigMap := .Values.controlPlane.extraConfigMaps }} + - name: {{ $extraConfigMap.name }} + mountPath: {{ $extraConfigMap.mountPath }} + readOnly: {{ $extraConfigMap.readOnly }} + {{- end }} + {{- range $mergedSecret := $mergedSecrets }} + - name: {{ $mergedSecret.name }} + mountPath: {{ $mergedSecret.mountPath }} + subPath: {{ $mergedSecret.subPath }} + readOnly: {{ $mergedSecret.readOnly }} + {{- end }} + - name: tmp + mountPath: /tmp + volumes: + {{- if eq .Values.controlPlane.environment "kubernetes" }} + {{- if not .Values.controlPlane.automountServiceAccountToken }} + - name: serviceaccount-token + projected: + defaultMode: 420 + sources: + - serviceAccountToken: + expirationSeconds: 3600 + path: token + - configMap: + name: kube-root-ca.crt + items: + - key: ca.crt + path: ca.crt + - downwardAPI: + items: + - fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + path: namespace + {{- end }} + {{- if .Values.controlPlane.tls.general.secretName }} + - name: general-tls-cert + secret: + secretName: {{ .Values.controlPlane.tls.general.secretName }} + {{- else }} + - name: general-tls-cert + secret: + secretName: {{ include "kuma.name" . }}-tls-cert + {{- end }} + {{- if .Values.controlPlane.tls.general.caSecretName }} + - name: general-tls-cert-ca + secret: + secretName: {{ .Values.controlPlane.tls.general.caSecretName }} + {{- end }} + {{- end }} + {{- if and (eq .Values.controlPlane.environment "universal") (eq .Values.controlPlane.mode "zone") }} + {{- if .Values.controlPlane.tls.general.secretName }} + - name: general-tls-cert + secret: + secretName: {{ .Values.controlPlane.tls.general.secretName }} + {{- end }} + {{- if .Values.controlPlane.tls.general.caSecretName }} + - name: general-tls-cert-ca + secret: + secretName: {{ .Values.controlPlane.tls.general.caSecretName }} + {{- end }} + {{- end }} + {{- if .Values.controlPlane.tls.apiServer.secretName }} + - name: api-server-tls-cert + secret: + secretName: {{ .Values.controlPlane.tls.apiServer.secretName }} + {{- end }} + {{- if .Values.postgres.tls.caSecretName }} + - name: postgres-tls-cert-ca + secret: + secretName: {{ .Values.postgres.tls.caSecretName }} + {{- end }} + {{- if .Values.postgres.tls.secretName }} + - name: postgres-tls-cert + secret: + secretName: {{ .Values.postgres.tls.secretName }} + {{- end }} + {{- if .Values.controlPlane.tls.apiServer.clientCertsSecretName }} + - name: api-server-client-certs + secret: + secretName: {{ .Values.controlPlane.tls.apiServer.clientCertsSecretName }} + {{- end }} + {{- if $kdsGlobalServerTLSSecretName }} + - name: kds-server-tls-cert + secret: + secretName: {{ $kdsGlobalServerTLSSecretName }} + {{- end }} + {{- if $kdsZoneClientTLSSecretName }} + - name: kds-client-tls-cert + secret: + secretName: {{ $kdsZoneClientTLSSecretName }} + {{- end }} + - name: {{ include "kuma.name" . }}-control-plane-config + configMap: + name: {{ include "kuma.name" . }}-control-plane-config + {{- range $extraConfigMap := .Values.controlPlane.extraConfigMaps }} + - name: {{ $extraConfigMap.name }} + configMap: + name: {{ $extraConfigMap.name }} + {{- end }} + {{- range $mergedSecret := $mergedSecrets }} + - name: {{ $mergedSecret.name }} + secret: + secretName: {{ $mergedSecret.name }} + {{- end }} + - name: tmp + emptyDir: {} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-global-sync-service.yaml b/charts/kuma/kuma/2.9.1/templates/cp-global-sync-service.yaml new file mode 100644 index 000000000..c5b3555a8 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-global-sync-service.yaml @@ -0,0 +1,33 @@ +{{- if and (eq .Values.controlPlane.mode "global") .Values.controlPlane.globalZoneSyncService.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "kuma.controlPlane.globalZoneSync.serviceName" . }} + namespace: {{ .Release.Namespace }} + annotations: + {{- range $key, $value := .Values.controlPlane.globalZoneSyncService.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +spec: + type: {{ .Values.controlPlane.globalZoneSyncService.type }} + {{- if .Values.controlPlane.globalZoneSyncService.loadBalancerIP }} + loadBalancerIP: {{ .Values.controlPlane.globalZoneSyncService.loadBalancerIP }} + {{- end }} + {{- if .Values.controlPlane.globalZoneSyncService.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range .Values.controlPlane.globalZoneSyncService.loadBalancerSourceRanges }} + - {{.}} + {{- end }} + {{- end }} + ports: + - port: {{ .Values.controlPlane.globalZoneSyncService.port }} + appProtocol: {{ .Values.controlPlane.globalZoneSyncService.protocol }} + {{- if and (eq .Values.controlPlane.globalZoneSyncService.type "NodePort") .Values.controlPlane.globalZoneSyncService.nodePort }} + nodePort: {{ .Values.controlPlane.globalZoneSyncService.nodePort }} + {{- end }} + name: global-zone-sync + selector: + app: {{ include "kuma.name" . }}-control-plane + {{ include "kuma.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-hpa.yaml b/charts/kuma/kuma/2.9.1/templates/cp-hpa.yaml new file mode 100644 index 000000000..dc4981020 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-hpa.yaml @@ -0,0 +1,24 @@ +{{- if .Values.controlPlane.autoscaling.enabled }} +{{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} +apiVersion: "autoscaling/v2" +{{ else }} +apiVersion: "autoscaling/v1" +{{ end }} +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "kuma.name" . }}-control-plane + minReplicas: {{ .Values.controlPlane.autoscaling.minReplicas }} + maxReplicas: {{ .Values.controlPlane.autoscaling.maxReplicas }} + {{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} + metrics: {{- toYaml .Values.controlPlane.autoscaling.metrics | nindent 4 }} + {{ else }} + targetCPUUtilizationPercentage: {{ .Values.controlPlane.autoscaling.targetCPUUtilizationPercentage }} + {{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-ingress.yaml b/charts/kuma/kuma/2.9.1/templates/cp-ingress.yaml new file mode 100644 index 000000000..8ceae01f8 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-ingress.yaml @@ -0,0 +1,25 @@ +{{- if .Values.controlPlane.ingress.enabled }} +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: {{ include "kuma.controlPlane.serviceName" . }} + namespace: {{ .Release.Namespace }} + {{- with .Values.controlPlane.ingress.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +spec: + ingressClassName: {{ .Values.controlPlane.ingress.ingressClassName }} + rules: + - host: {{ .Values.controlPlane.ingress.hostname }} + http: + paths: + - path: {{ .Values.controlPlane.ingress.path }} + pathType: {{ .Values.controlPlane.ingress.pathType }} + backend: + service: + name: {{ include "kuma.controlPlane.serviceName" . }} + port: + number: {{ .Values.controlPlane.ingress.servicePort }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-kds-global-server-secret.yaml b/charts/kuma/kuma/2.9.1/templates/cp-kds-global-server-secret.yaml new file mode 100644 index 000000000..5ea3314a3 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-kds-global-server-secret.yaml @@ -0,0 +1,15 @@ +{{ if and (eq .Values.controlPlane.mode "global") .Values.controlPlane.tls.kdsGlobalServer.create }} +apiVersion: v1 +kind: Secret +metadata: +{{ with .Values.controlPlane.tls.kdsGlobalServer.secretName }} + name: {{ . }} +{{ else }} + name: {{ include "kuma.name" . }}-kds-global-server-tls +{{ end }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +type: kubernetes.io/tls +stringData: + tls.crt: {{ required "you must provide a kds tls cert" .Values.controlPlane.tls.kdsGlobalServer.cert | quote }} + tls.key: {{ required "you must provide a kds tls key" .Values.controlPlane.tls.kdsGlobalServer.key | quote }} +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-kds-zone-client-tls-secret.yaml b/charts/kuma/kuma/2.9.1/templates/cp-kds-zone-client-tls-secret.yaml new file mode 100644 index 000000000..99b15c5bd --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-kds-zone-client-tls-secret.yaml @@ -0,0 +1,13 @@ +{{ if and (eq .Values.controlPlane.mode "zone") .Values.controlPlane.tls.kdsZoneClient.create }} +apiVersion: v1 +kind: Secret +metadata: +{{ with .Values.controlPlane.tls.kdsZoneClient.secretName }} + name: {{ . }} +{{ else }} + name: {{ include "kuma.name" . }}-kds-zone-client-tls +{{ end }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +stringData: + ca.crt: {{ required "you must provide a kds cert" .Values.controlPlane.tls.kdsZoneClient.cert | quote }} +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-pdb.yaml b/charts/kuma/kuma/2.9.1/templates/cp-pdb.yaml new file mode 100644 index 000000000..bb29bfd20 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-pdb.yaml @@ -0,0 +1,20 @@ +{{ if $.Values.controlPlane.podDisruptionBudget.enabled }} +{{ if .Capabilities.APIVersions.Has "policy/v1" }} +apiVersion: policy/v1 +{{ else if .Capabilities.APIVersions.Has "policy/v1beta1" }} +apiVersion: policy/v1beta1 +{{ else }} +{{ fail "pod disruption budgets are not supported by this version of kubernetes" }} +{{ end }} +kind: PodDisruptionBudget +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.controlPlane.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-control-plane +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-rbac.yaml b/charts/kuma/kuma/2.9.1/templates/cp-rbac.yaml new file mode 100644 index 000000000..52ce1bfa8 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-rbac.yaml @@ -0,0 +1,320 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +{{- with .Values.controlPlane.serviceAccountAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} +{{- end }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +{{- if (eq .Values.controlPlane.environment "kubernetes") }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-control-plane + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - namespaces + - pods +{{- if not (and .Values.transparentProxy.configMap.enabled .Values.transparentProxy.configMap.config) }} + - configmaps +{{- end }} + - nodes +{{- if .Values.controlPlane.supportGatewaySecretsInAllNamespaces }} + - secrets +{{- end }} + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - secrets + verbs: + - list + - watch + - apiGroups: + - "discovery.k8s.io" + resources: + - endpointslices + verbs: + - get + - list + - watch + - apiGroups: + - "apps" + resources: + - deployments + - replicasets + verbs: + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - "batch" + resources: + - jobs + verbs: + - get + - list + - watch + - apiGroups: + - gateway.networking.k8s.io + resources: + - gatewayclasses + - gateways + - referencegrants + - httproutes + verbs: + - create + - delete + - get + - list + - patch + - update + - watch + - apiGroups: + - gateway.networking.k8s.io + resources: + - gatewayclasses/status + - gateways/status + - httproutes/status + verbs: + - get + - patch + - update + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch + - apiGroups: + - "" + resources: + - services +{{- if and .Values.transparentProxy.configMap.enabled .Values.transparentProxy.configMap.config }} + - configmaps +{{- end }} + verbs: + - get + - delete + - list + - watch + - create + - update + - patch + - apiGroups: + - "discovery.k8s.io" + resources: + - endpointslices + verbs: + - get + - list + - watch + - apiGroups: + - kuma.io + resources: + - dataplanes + - dataplaneinsights + - meshes + - zones + - zoneinsights + - zoneingresses + - zoneingressinsights + - zoneegresses + - zoneegressinsights + - meshinsights + - serviceinsights + - proxytemplates + - ratelimits + - trafficpermissions + - trafficroutes + - timeouts + - retries + - circuitbreakers + - virtualoutbounds + - containerpatches + - externalservices + - faultinjections + - healthchecks + - trafficlogs + - traffictraces + - meshgateways + - meshgatewayroutes + - meshgatewayinstances + - meshgatewayconfigs + {{- range $policy, $v := .Values.plugins.policies }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{- range $policy, $v := .Values.plugins.resources }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - kuma.io + resources: + - meshgatewayinstances/status + - meshgatewayinstances/finalizers + - meshes/finalizers + - dataplanes/finalizers + verbs: + - get + - patch + - update + - apiGroups: + - "" + resources: + - pods/finalizers + verbs: + - get + - patch + - update + {{- if .Values.cni.enabled }} + - apiGroups: + - k8s.cni.cncf.io + resources: + - network-attachment-definitions + verbs: + - get + - list + - watch + - create + - delete + - apiGroups: + - apiextensions.k8s.io + resources: + - customresourcedefinitions + verbs: + - get + - list + - watch + - apiGroups: + - "" + resources: + - nodes + verbs: + - update + - apiGroups: + - "pods" + resources: + - pods + verbs: + - list + {{- end }} + # validate k8s token before issuing mTLS cert + - apiGroups: + - authentication.k8s.io + resources: + - tokenreviews + verbs: + - create +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-control-plane + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-control-plane +subjects: + - kind: ServiceAccount + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + # leader-for-life election deletes Pods in some circumstances + - apiGroups: + - "" + resources: + - pods + verbs: + - delete +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "kuma.name" . }}-control-plane +subjects: + - kind: ServiceAccount + name: {{ include "kuma.name" . }}-control-plane + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-service.yaml b/charts/kuma/kuma/2.9.1/templates/cp-service.yaml new file mode 100644 index 000000000..3b9c3e31f --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-service.yaml @@ -0,0 +1,49 @@ +{{ if .Values.controlPlane.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "kuma.controlPlane.serviceName" . }} + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} + annotations: + {{- range $key, $value := .Values.controlPlane.service.annotations }} + {{- if $value }} + {{ $key }}: {{ $value | quote }} + {{- end }} + {{- end }} +spec: + type: {{ .Values.controlPlane.service.type }} + ports: + - port: 5680 + name: diagnostics + appProtocol: http + - port: 5681 + name: http-api-server + appProtocol: http + {{- if and (eq .Values.controlPlane.service.type "NodePort") .Values.controlPlane.service.apiServer.http.nodePort }} + nodePort: {{ .Values.controlPlane.service.apiServer.http.nodePort }} + {{- end }} + - port: 5682 + name: https-api-server + appProtocol: https + {{- if and (eq .Values.controlPlane.service.type "NodePort") .Values.controlPlane.service.apiServer.https.nodePort }} + nodePort: {{ .Values.controlPlane.service.apiServer.https.nodePort }} + {{- end }} + {{- if ne .Values.controlPlane.environment "universal" }} + - port: 443 + name: https-admission-server + targetPort: {{ .Values.controlPlane.admissionServerPort | default "5443" }} + appProtocol: https + {{- end }} + {{- if ne .Values.controlPlane.mode "global" }} + - port: 5676 + name: mads-server + appProtocol: https + - port: 5678 + name: dp-server + appProtocol: https + {{- end }} + selector: + app: {{ include "kuma.name" . }}-control-plane + {{- include "kuma.selectorLabels" . | nindent 4 }} +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/cp-webhooks-and-secrets.yaml b/charts/kuma/kuma/2.9.1/templates/cp-webhooks-and-secrets.yaml new file mode 100644 index 000000000..15b38e0fd --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/cp-webhooks-and-secrets.yaml @@ -0,0 +1,346 @@ +{{- if not (eq (empty .Values.controlPlane.tls.general.caBundle) (empty .Values.controlPlane.tls.general.secretName)) }} + {{ fail "You need to send both or neither of controlPlane.tls.general.caBundle and controlPlane.tls.general.secretName"}} +{{- end }} +{{- $caBundle := .Values.controlPlane.tls.general.caBundle }} +{{/* +Generate certificates +see: https://masterminds.github.io/sprig/crypto.html +see: https://medium.com/nuvo-group-tech/move-your-certs-to-helm-4f5f61338aca +see: https://github.com/networkservicemesh/networkservicemesh/blob/804ad5026bb5dbd285c220f15395fe25e46f5edb/deployments/helm/nsm/charts/admission-webhook/templates/admission-webhook-secret.tpl + +We only autogenerate certs if user did not chose their own secret. +We only autogenerate certs if the cert is not yet generated. This way we keep the secrets between HELM upgrades. +*/}} + +{{- if eq .Values.controlPlane.tls.general.secretName "" -}} +{{- $cert := "" }} +{{- $key := "" }} +{{- $secretName := print (include "kuma.name" .) "-tls-cert" }} + +{{- $secret := (lookup "v1" "Secret" .Release.Namespace $secretName) -}} +{{- if $secret -}} + {{- $cert = index $secret.data "tls.crt" -}} + {{- $key = index $secret.data "tls.key" -}} + {{- $caBundle = index $secret.data "ca.crt" -}} +{{- else -}} + {{- $commonName := (include "kuma.controlPlane.serviceName" .) -}} + {{- $altNames := list (printf "%s.%s" $commonName .Release.Namespace) (printf "%s.%s.svc" $commonName .Release.Namespace) -}} + {{- $certTTL := 3650 -}} + {{- $ca := genCA "kuma-ca" $certTTL -}} + + {{- $genCert := genSignedCert $commonName nil $altNames $certTTL $ca -}} + {{- $cert = $genCert.Cert | b64enc -}} + {{- $key = $genCert.Key | b64enc -}} + {{ $caBundle = $ca.Cert | b64enc }} +{{- end -}} +--- +apiVersion: v1 +kind: Secret +type: kubernetes.io/tls +metadata: + name: {{ $secretName }} + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +data: + tls.crt: {{ $cert }} + tls.key: {{ $key }} + ca.crt: {{ $caBundle }} +{{- end }} +{{- if (eq .Values.controlPlane.environment "kubernetes") }} +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ include "kuma.name" . }}-admission-mutating-webhook-configuration + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +webhooks: + - name: mesh.defaulter.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: Fail + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /default-kuma-io-v1alpha1-mesh + rules: + - apiGroups: + - kuma.io + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - meshes + - dataplanes + - dataplaneinsights + - meshgateways + - zoneingresses + - zoneingressinsights + - zoneegresses + - zoneegressinsights + - serviceinsights + - zone + - zoneinsights + {{- range $policy, $v := .Values.plugins.policies }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{- range $policy, $v := .Values.plugins.resources }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + sideEffects: None + - name: owner-reference.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: Fail + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /owner-reference-kuma-io-v1alpha1 + rules: + - apiGroups: + - kuma.io + apiVersions: + - v1alpha1 + operations: + - CREATE + resources: + - circuitbreakers + - externalservices + - faultinjections + - healthchecks + - meshgateways + - meshgatewayroutes + - proxytemplates + - ratelimits + - retries + - timeouts + - trafficlogs + - trafficpermissions + - trafficroutes + - traffictraces + - virtualoutbounds + {{- range $policy, $v := .Values.plugins.policies }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{- range $policy, $v := .Values.plugins.resources }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{ .Values.controlPlane.webhooks.ownerReference.additionalRules | nindent 6 }} + sideEffects: None + {{- if ne .Values.controlPlane.mode "global" }} + - name: namespace-kuma-injector.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: {{ .Values.controlPlane.injectorFailurePolicy }} + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + - key: kuma.io/sidecar-injection + operator: In + values: ["enabled", "true"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /inject-sidecar + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + resources: + - pods + sideEffects: None + - name: pods-kuma-injector.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: {{ .Values.controlPlane.injectorFailurePolicy }} + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + objectSelector: + matchLabels: + kuma.io/sidecar-injection: enabled + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /inject-sidecar + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + resources: + - pods + sideEffects: None + {{- end }} +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: {{ include "kuma.name" . }}-validating-webhook-configuration + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.cpLabels" . | nindent 4 }} +webhooks: + - name: validator.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: Fail + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /validate-kuma-io-v1alpha1 + rules: + - apiGroups: + - kuma.io + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + - DELETE + resources: + - circuitbreakers + - dataplanes + - externalservices + - faultinjections + - meshgatewayinstances + - healthchecks + - meshes + - meshgateways + - meshgatewayroutes + - proxytemplates + - ratelimits + - retries + - trafficlogs + - trafficpermissions + - trafficroutes + - traffictraces + - virtualoutbounds + - zones + - containerpatches + {{- range $policy, $v := .Values.plugins.policies }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{- range $policy, $v := .Values.plugins.resources }} + {{- if $v }} + - {{ $policy }} + {{- end}} + {{- end}} + {{ .Values.controlPlane.webhooks.validator.additionalRules | nindent 6 }} + sideEffects: None + {{- if ne .Values.controlPlane.mode "global" }} + - name: service.validator.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: Ignore + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /validate-v1-service + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - services + sideEffects: None + {{- end }} + - name: secret.validator.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + namespaceSelector: + matchLabels: + kuma.io/system-namespace: "true" + failurePolicy: Ignore + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /validate-v1-secret + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + - DELETE + resources: + - secrets + sideEffects: None + - name: gateway.validator.kuma-admission.kuma.io + admissionReviewVersions: ["v1"] + failurePolicy: Ignore + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: ["kube-system"] + clientConfig: + caBundle: {{ $caBundle }} + service: + namespace: {{ .Release.Namespace }} + name: {{ include "kuma.controlPlane.serviceName" . }} + path: /validate-gatewayclass + rules: + - apiGroups: + - "gateway.networking.k8s.io" + apiVersions: + - v1beta1 + operations: + - CREATE + resources: + - gatewayclasses + sideEffects: None +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/egress-deployment.yaml b/charts/kuma/kuma/2.9.1/templates/egress-deployment.yaml new file mode 100644 index 000000000..3b6617eee --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/egress-deployment.yaml @@ -0,0 +1,138 @@ +{{- if .Values.egress.enabled }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "kuma.name" . }}-egress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.egressLabels" . | nindent 4 }} +spec: + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 0 + {{- if not .Values.egress.autoscaling.enabled }} + replicas: {{ .Values.egress.replicas }} + {{- end }} + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-egress + template: + metadata: + annotations: + kuma.io/egress: enabled + {{- range $key, $value := merge .Values.egress.podAnnotations .Values.egress.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: + {{- include "kuma.egressLabels" . | nindent 8 }} + spec: + {{- with .Values.egress.affinity }} + affinity: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + {{- with .Values.egress.topologySpreadConstraints }} + topologySpreadConstraints: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + securityContext: + {{- toYaml .Values.egress.podSecurityContext | trim | nindent 8 }} + serviceAccountName: {{ include "kuma.name" . }}-egress + automountServiceAccountToken: {{ .Values.egress.automountServiceAccountToken }} + {{- with .Values.egress.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.egress.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + {{ include "kuma.dnsConfig" (dict "dns" .Values.egress.dns) | nindent 6 | trim }} + containers: + - name: egress + image: {{ include "kuma.formatImage" (dict "image" .Values.dataPlane.image "root" $) | quote }} + imagePullPolicy: {{ .Values.dataPlane.image.pullPolicy }} + securityContext: + {{- toYaml .Values.egress.containerSecurityContext | trim | nindent 12 }} + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: KUMA_CONTROL_PLANE_URL + value: "https://{{ include "kuma.controlPlane.serviceName" . }}.{{ .Release.Namespace }}:5678" + - name: KUMA_CONTROL_PLANE_CA_CERT_FILE + value: /var/run/secrets/kuma.io/cp-ca/ca.crt + - name: KUMA_DATAPLANE_DRAIN_TIME + value: {{ .Values.egress.drainTime }} + - name: KUMA_DATAPLANE_RUNTIME_TOKEN_PATH + value: /var/run/secrets/kubernetes.io/serviceaccount/token + - name: KUMA_DATAPLANE_PROXY_TYPE + value: "egress" + args: + - run + - --log-level={{ .Values.egress.logLevel | default "info" }} + ports: + - containerPort: 10002 + livenessProbe: + httpGet: + path: "/ready" + port: 9901 + failureThreshold: 12 + initialDelaySeconds: 60 + periodSeconds: 5 + successThreshold: 1 + timeoutSeconds: 3 + readinessProbe: + httpGet: + path: "/ready" + port: 9901 + failureThreshold: 12 + initialDelaySeconds: 1 + periodSeconds: 5 + successThreshold: 1 + timeoutSeconds: 3 + resources: {{ toYaml .Values.egress.resources | nindent 12 }} + volumeMounts: +{{- if not .Values.egress.automountServiceAccountToken }} + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: serviceaccount-token + readOnly: true +{{- end }} + - name: control-plane-ca + mountPath: /var/run/secrets/kuma.io/cp-ca + readOnly: true + - name: tmp + mountPath: /tmp + volumes: +{{- if not .Values.egress.automountServiceAccountToken }} + - name: serviceaccount-token + projected: + defaultMode: 420 + sources: + - serviceAccountToken: + expirationSeconds: 3600 + path: token + - configMap: + name: kube-root-ca.crt + items: + - key: ca.crt + path: ca.crt + - downwardAPI: + items: + - fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + path: namespace +{{- end }} + - name: control-plane-ca + secret: + secretName: {{ include "kuma.controlPlane.tls.general.caSecretName" . }} + items: + - key: ca.crt + path: ca.crt + - name: tmp + emptyDir: {} + {{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/egress-hpa.yaml b/charts/kuma/kuma/2.9.1/templates/egress-hpa.yaml new file mode 100644 index 000000000..8d4284f41 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/egress-hpa.yaml @@ -0,0 +1,24 @@ +{{- if .Values.egress.autoscaling.enabled }} +{{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} +apiVersion: "autoscaling/v2" +{{ else }} +apiVersion: "autoscaling/v1" +{{ end }} +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "kuma.name" . }}-egress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.egressLabels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "kuma.name" . }}-egress + minReplicas: {{ .Values.egress.autoscaling.minReplicas }} + maxReplicas: {{ .Values.egress.autoscaling.maxReplicas }} + {{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} + metrics: {{- toYaml .Values.egress.autoscaling.metrics | nindent 4 }} + {{ else }} + targetCPUUtilizationPercentage: {{ .Values.egress.autoscaling.targetCPUUtilizationPercentage }} + {{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/egress-pdb.yaml b/charts/kuma/kuma/2.9.1/templates/egress-pdb.yaml new file mode 100644 index 000000000..ee599003b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/egress-pdb.yaml @@ -0,0 +1,20 @@ +{{ if $.Values.egress.podDisruptionBudget.enabled }} +{{ if .Capabilities.APIVersions.Has "policy/v1" }} +apiVersion: policy/v1 +{{ else if .Capabilities.APIVersions.Has "policy/v1beta1" }} +apiVersion: policy/v1beta1 +{{ else }} +{{ fail "pod disruption budgets are not supported by this version of kubernetes" }} +{{ end }} +kind: PodDisruptionBudget +metadata: + name: {{ include "kuma.name" . }}-egress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.egressLabels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.egress.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-egress +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/egress-rbac.yaml b/charts/kuma/kuma/2.9.1/templates/egress-rbac.yaml new file mode 100644 index 000000000..1b4326fdb --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/egress-rbac.yaml @@ -0,0 +1,18 @@ +{{- if .Values.egress.enabled }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "kuma.name" . }}-egress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.egressLabels" . | nindent 4 }} +{{- with .Values.egress.serviceAccountAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} +{{- end }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/egress-service.yaml b/charts/kuma/kuma/2.9.1/templates/egress-service.yaml new file mode 100644 index 000000000..2127811fe --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/egress-service.yaml @@ -0,0 +1,32 @@ +{{- if .Values.egress.enabled }} +{{- if eq .Values.controlPlane.mode "global" }} +{{ fail "You shouldn't run zoneEgress when running the CP in global" }} +{{- end }} +{{- end }} +{{- if and .Values.egress.enabled .Values.egress.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "kuma.egress.serviceName" . }} + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.egressLabels" . | nindent 4 }} + annotations: + {{- range $key, $value := .Values.egress.service.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} +spec: + type: {{ .Values.egress.service.type }} + {{- if .Values.egress.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.egress.service.loadBalancerIP }} + {{- end }} + ports: + - port: {{ .Values.egress.service.port }} + protocol: TCP + targetPort: 10002 + {{- if and (eq .Values.egress.service.type "NodePort") .Values.egress.service.nodePort }} + nodePort: {{ .Values.egress.service.nodePort }} + {{- end }} + selector: + app: {{ include "kuma.name" . }}-egress + {{- include "kuma.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/gateway-class.yaml b/charts/kuma/kuma/2.9.1/templates/gateway-class.yaml new file mode 100644 index 000000000..cf1ae305d --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/gateway-class.yaml @@ -0,0 +1,19 @@ +{{- if and (eq .Values.controlPlane.environment "kubernetes") (eq .Values.controlPlane.mode "zone") }} +{{- if .Capabilities.APIVersions.Has "gateway.networking.k8s.io/v1/GatewayClass" }} +--- +apiVersion: gateway.networking.k8s.io/v1 +kind: GatewayClass +metadata: + name: kuma +spec: + controllerName: "gateways.kuma.io/controller" +{{- else if .Capabilities.APIVersions.Has "gateway.networking.k8s.io/v1beta1/GatewayClass" }} +--- +apiVersion: gateway.networking.k8s.io/v1beta1 +kind: GatewayClass +metadata: + name: kuma +spec: + controllerName: "gateways.kuma.io/controller" +{{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/ingress-deployment.yaml b/charts/kuma/kuma/2.9.1/templates/ingress-deployment.yaml new file mode 100644 index 000000000..fcefeaac6 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/ingress-deployment.yaml @@ -0,0 +1,142 @@ +{{- if .Values.ingress.enabled }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "kuma.name" . }}-ingress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.ingressLabels" . | nindent 4 }} +spec: + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 0 + {{- if not .Values.ingress.autoscaling.enabled }} + replicas: {{ .Values.ingress.replicas }} + {{- end }} + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-ingress + template: + metadata: + annotations: + kuma.io/ingress: enabled + {{- range $key, $value := merge .Values.ingress.podAnnotations .Values.ingress.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: + {{- include "kuma.ingressLabels" . | nindent 8 }} + spec: + {{- with .Values.ingress.affinity }} + affinity: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + {{- with .Values.ingress.topologySpreadConstraints }} + topologySpreadConstraints: {{ tpl (toYaml . | nindent 8) $ }} + {{- end }} + securityContext: + {{- toYaml .Values.ingress.podSecurityContext | trim | nindent 8 }} + serviceAccountName: {{ include "kuma.name" . }}-ingress + automountServiceAccountToken: {{ .Values.ingress.automountServiceAccountToken }} + {{- with .Values.ingress.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.ingress.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.ingress.terminationGracePeriodSeconds }} + {{ include "kuma.dnsConfig" (dict "dns" .Values.ingress.dns) | nindent 6 | trim }} + containers: + - name: ingress + image: {{ include "kuma.formatImage" (dict "image" .Values.dataPlane.image "root" $) | quote }} + imagePullPolicy: {{ .Values.dataPlane.image.pullPolicy }} + securityContext: + {{- toYaml .Values.ingress.containerSecurityContext | trim | nindent 12 }} + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: KUMA_CONTROL_PLANE_URL + value: "https://{{ include "kuma.controlPlane.serviceName" . }}.{{ .Release.Namespace }}:5678" + - name: KUMA_CONTROL_PLANE_CA_CERT_FILE + value: /var/run/secrets/kuma.io/cp-ca/ca.crt + - name: KUMA_DATAPLANE_DRAIN_TIME + value: {{ .Values.ingress.drainTime }} + - name: KUMA_DATAPLANE_RUNTIME_TOKEN_PATH + value: /var/run/secrets/kubernetes.io/serviceaccount/token + - name: KUMA_DATAPLANE_PROXY_TYPE + value: "ingress" + args: + - run + - --log-level={{ .Values.ingress.logLevel | default "info" }} + ports: + - containerPort: 10001 + livenessProbe: + httpGet: + path: "/ready" + port: 9901 + failureThreshold: 12 + initialDelaySeconds: 60 + periodSeconds: 5 + successThreshold: 1 + timeoutSeconds: 3 + readinessProbe: + httpGet: + path: "/ready" + port: 9901 + failureThreshold: 12 + initialDelaySeconds: 1 + periodSeconds: 5 + successThreshold: 1 + timeoutSeconds: 3 + resources: {{ toYaml .Values.ingress.resources | nindent 12 }} + {{ with .Values.ingress.lifecycle}} + lifecycle: {{ . | toYaml | nindent 12 }} + {{ end }} + volumeMounts: +{{- if not .Values.ingress.automountServiceAccountToken }} + - mountPath: /var/run/secrets/kubernetes.io/serviceaccount + name: serviceaccount-token + readOnly: true +{{- end }} + - name: control-plane-ca + mountPath: /var/run/secrets/kuma.io/cp-ca + readOnly: true + - name: tmp + mountPath: /tmp + volumes: +{{- if not .Values.ingress.automountServiceAccountToken }} + - name: serviceaccount-token + projected: + defaultMode: 420 + sources: + - serviceAccountToken: + expirationSeconds: 3600 + path: token + - configMap: + name: kube-root-ca.crt + items: + - key: ca.crt + path: ca.crt + - downwardAPI: + items: + - fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + path: namespace +{{- end }} + - name: control-plane-ca + secret: + secretName: {{ include "kuma.controlPlane.tls.general.caSecretName" . }} + items: + - key: ca.crt + path: ca.crt + - name: tmp + emptyDir: {} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/ingress-hpa.yaml b/charts/kuma/kuma/2.9.1/templates/ingress-hpa.yaml new file mode 100644 index 000000000..4aaeabe67 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/ingress-hpa.yaml @@ -0,0 +1,24 @@ +{{- if .Values.ingress.autoscaling.enabled }} +{{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} +apiVersion: "autoscaling/v2" +{{ else }} +apiVersion: "autoscaling/v1" +{{ end }} +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "kuma.name" . }}-ingress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.ingressLabels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "kuma.name" . }}-ingress + minReplicas: {{ .Values.ingress.autoscaling.minReplicas }} + maxReplicas: {{ .Values.ingress.autoscaling.maxReplicas }} + {{ if .Capabilities.APIVersions.Has "autoscaling/v2" }} + metrics: {{- toYaml .Values.ingress.autoscaling.metrics | nindent 4 }} + {{ else }} + targetCPUUtilizationPercentage: {{ .Values.ingress.autoscaling.targetCPUUtilizationPercentage }} + {{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/ingress-pdb.yaml b/charts/kuma/kuma/2.9.1/templates/ingress-pdb.yaml new file mode 100644 index 000000000..639d1b574 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/ingress-pdb.yaml @@ -0,0 +1,20 @@ +{{ if $.Values.ingress.podDisruptionBudget.enabled }} +{{ if .Capabilities.APIVersions.Has "policy/v1" }} +apiVersion: policy/v1 +{{ else if .Capabilities.APIVersions.Has "policy/v1beta1" }} +apiVersion: policy/v1beta1 +{{ else }} +{{ fail "pod disruption budgets are not supported by this version of kubernetes" }} +{{ end }} +kind: PodDisruptionBudget +metadata: + name: {{ include "kuma.name" . }}-ingress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.ingressLabels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.ingress.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "kuma.selectorLabels" . | nindent 6 }} + app: {{ include "kuma.name" . }}-ingress +{{ end }} diff --git a/charts/kuma/kuma/2.9.1/templates/ingress-rbac.yaml b/charts/kuma/kuma/2.9.1/templates/ingress-rbac.yaml new file mode 100644 index 000000000..e4e1d61ce --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/ingress-rbac.yaml @@ -0,0 +1,18 @@ +{{- if .Values.ingress.enabled }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "kuma.name" . }}-ingress + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.ingressLabels" . | nindent 4 }} +{{- with .Values.ingress.serviceAccountAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} +{{- end }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/ingress-service.yaml b/charts/kuma/kuma/2.9.1/templates/ingress-service.yaml new file mode 100644 index 000000000..74a4dde90 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/ingress-service.yaml @@ -0,0 +1,32 @@ +{{- if .Values.ingress.enabled }} +{{- if or (eq .Values.controlPlane.mode "global") (eq .Values.controlPlane.mode "standalone") }} +{{ fail "You shouldn't run zoneIngress when running the CP in global or standalone" }} +{{- end }} +{{- end }} +{{- if and .Values.ingress.enabled .Values.ingress.service.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "kuma.ingress.serviceName" . }} + namespace: {{ .Release.Namespace }} + labels: {{ include "kuma.ingressLabels" . | nindent 4 }} + annotations: + {{- range $key, $value := .Values.ingress.service.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} +spec: + type: {{ .Values.ingress.service.type }} + {{- if .Values.ingress.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.ingress.service.loadBalancerIP }} + {{- end }} + ports: + - port: {{ .Values.ingress.service.port }} + protocol: TCP + targetPort: 10001 + {{- if and (eq .Values.ingress.service.type "NodePort") .Values.ingress.service.nodePort }} + nodePort: {{ .Values.ingress.service.nodePort }} + {{- end }} + selector: + app: {{ include "kuma.name" . }}-ingress + {{- include "kuma.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/post-delete-cleanup-ebpf-job.yaml b/charts/kuma/kuma/2.9.1/templates/post-delete-cleanup-ebpf-job.yaml new file mode 100644 index 000000000..aaa3166ff --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/post-delete-cleanup-ebpf-job.yaml @@ -0,0 +1,126 @@ +{{- if and (.Values.experimental.ebpf.enabled) (and (not .Values.cni.enabled) (not .Values.noHelmHooks) (eq .Values.controlPlane.environment "kubernetes")) }} + {{- $serviceAccountName := printf "%s-cleanup-node-ebpf-job" (include "kuma.name" .) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "post-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-cleanup-node-ebpf-job + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "post-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +rules: + - apiGroups: [""] + resources: + - nodes + verbs: + - list + - apiGroups: [""] + resources: + - pods + verbs: + - watch + - delete + - deletecollection + - apiGroups: ["batch"] + resources: + - jobs + verbs: + - watch + - create + - delete + - deletecollection +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-cleanup-node-ebpf-job + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "post-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-cleanup-node-ebpf-job +subjects: + - kind: ServiceAccount + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "kuma.name" . }}-cleanup-node-ebpf-job + namespace: {{ .Release.Namespace }} + labels: + {{ include "kuma.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": "post-delete" + {{/* Ensure the job is created after the RBAC resources */}} + "helm.sh/hook-weight": "5" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" +spec: + template: + metadata: + name: {{ template "kuma.name" . }}-cleanup-node-ebpf-job + labels: + {{ include "kuma.labels" . | nindent 8 }} + spec: + serviceAccountName: {{ $serviceAccountName }} + {{- with .Values.hooks.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.hooks.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + {{- if .Values.hooks.ebpfCleanup.podSecurityContext }} + securityContext: + {{ toYaml .Values.hooks.ebpfCleanup.podSecurityContext | trim | nindent 8 }} + {{- end }} + containers: + - name: post-delete-job + image: {{ include "kuma.formatImage" (dict "image" .Values.dataPlane.initImage "root" $) | quote }} + {{- if .Values.hooks.ebpfCleanup.containerSecurityContext }} + securityContext: + {{ toYaml .Values.hooks.ebpfCleanup.containerSecurityContext | trim | nindent 12 }} + {{- end }} + resources: + requests: + cpu: "20m" + memory: "20Mi" + limits: + cpu: "40m" + memory: "40Mi" + command: + - 'kumactl' + - 'uninstall' + - 'ebpf' + - '--cleanup-image-registry' + - {{ .Values.global.image.registry }} + - '--cleanup-image-repository' + - {{ .Values.dataPlane.initImage.repository }} + {{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/pre-delete-webhooks.yaml b/charts/kuma/kuma/2.9.1/templates/pre-delete-webhooks.yaml new file mode 100644 index 000000000..e6948af2f --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/pre-delete-webhooks.yaml @@ -0,0 +1,109 @@ +{{- if and (eq .Values.controlPlane.environment "kubernetes") (not .Values.noHelmHooks) }} +# HELM first deletes RBAC of Kuma, then it tries to delete Secrets. We've got validating webhook on Secrets. +# But even that the policy of this webhook is Ignore, it fails because Kuma does not have permission to access Secrets anymore. +# Therefore we first need to delete webhook so we can delete the rest of the deployment +{{- $serviceAccountName := printf "%s-pre-delete-job" (include "kuma.name" .) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "pre-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-pre-delete-job + annotations: + "helm.sh/hook": "pre-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +rules: + - apiGroups: + - admissionregistration.k8s.io + resources: + - validatingwebhookconfigurations + resourceNames: + - {{ include "kuma.name" . }}-validating-webhook-configuration + verbs: + - delete +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-pre-delete-job + annotations: + "helm.sh/hook": "pre-delete" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-pre-delete-job +subjects: + - kind: ServiceAccount + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "kuma.name" . }}-delete-webhook + namespace: {{ .Release.Namespace }} + labels: + {{ include "kuma.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": "pre-delete" + {{/* Ensure the job is created after the RBAC resources */}} + "helm.sh/hook-weight": "5" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" +spec: + template: + metadata: + name: {{ template "kuma.name" . }}-delete-webhook + labels: + {{ include "kuma.labels" . | nindent 8 }} + spec: + serviceAccountName: {{ $serviceAccountName }} + {{- with .Values.hooks.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.hooks.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + securityContext: + {{- toYaml .Values.hooks.podSecurityContext | trim | nindent 8 }} + containers: + - name: pre-delete-job + image: "{{ .Values.kubectl.image.registry }}/{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" + command: + - 'kubectl' + - 'delete' + - 'ValidatingWebhookConfiguration' + - '--ignore-not-found' + - {{ include "kuma.name" . }}-validating-webhook-configuration + securityContext: + {{- toYaml (mergeOverwrite (dict "runAsUser" 65534) .Values.hooks.containerSecurityContext) | trim | nindent 12 }} + resources: + requests: + cpu: "100m" + memory: "256Mi" + limits: + cpu: "100m" + memory: "256Mi" +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/pre-install-patch-namespace-job.yaml b/charts/kuma/kuma/2.9.1/templates/pre-install-patch-namespace-job.yaml new file mode 100644 index 000000000..a84d7accf --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/pre-install-patch-namespace-job.yaml @@ -0,0 +1,124 @@ +{{- if and ( .Values.noHelmHooks ) (eq .Values.controlPlane.environment "kubernetes") }} + {{- $errorMessage := ".Values.noHelmHooks is set. You must manually create and label the system namespace with kuma.io/system-namespace: \"true\" before installing or upgrading the chart" }} + {{- $systemNamespace := (lookup "v1" "Namespace" "" .Release.Namespace) }} + {{- if not $systemNamespace }} + {{- fail $errorMessage }} + {{- end }} + {{- $systemNamespaceLabels := ($systemNamespace).metadata.labels }} + {{- if ne (get $systemNamespaceLabels "kuma.io/system-namespace") "true" }} + {{- fail $errorMessage }} + {{- end }} +{{- else}} + {{- if .Values.patchSystemNamespace }} + {{- $serviceAccountName := printf "%s-patch-ns-job" (include "kuma.name" .) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +{{- with .Values.global.imagePullSecrets }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-patch-ns-job + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - namespaces + resourceNames: + - {{ .Release.Namespace }} + verbs: + - get + - patch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-patch-ns-job + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "pre-install" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-patch-ns-job +subjects: + - kind: ServiceAccount + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "kuma.name" . }}-patch-ns + namespace: {{ .Release.Namespace }} + labels: + {{ include "kuma.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": "pre-install" + {{/* Ensure the job is created after the RBAC resources */}} + "helm.sh/hook-weight": "5" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" +spec: + template: + metadata: + name: {{ template "kuma.name" . }}-patch-ns-script + labels: + {{ include "kuma.labels" . | nindent 8 }} + spec: + serviceAccountName: {{ $serviceAccountName }} + {{- with .Values.hooks.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.hooks.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + securityContext: + {{- toYaml .Values.hooks.podSecurityContext | trim | nindent 8 }} + containers: + - name: pre-install-job + image: "{{ .Values.kubectl.image.registry }}/{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" + securityContext: + {{- toYaml (mergeOverwrite (dict "runAsUser" 65534) .Values.hooks.containerSecurityContext) | trim | nindent 12 }} + resources: + requests: + cpu: "100m" + memory: "256Mi" + limits: + cpu: "100m" + memory: "256Mi" + command: + - 'kubectl' + - 'patch' + - 'namespace' + - {{ .Release.Namespace | quote }} + - '--type' + - 'merge' + - '--patch' + - '{ "metadata": { "labels": { "kuma.io/system-namespace": "true" } } }' + {{- end }} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/templates/pre-upgrade-install-crds-job.yaml b/charts/kuma/kuma/2.9.1/templates/pre-upgrade-install-crds-job.yaml new file mode 100644 index 000000000..8fadf1722 --- /dev/null +++ b/charts/kuma/kuma/2.9.1/templates/pre-upgrade-install-crds-job.yaml @@ -0,0 +1,171 @@ +{{- if (and .Values.installCrdsOnUpgrade.enabled (and (not .Values.noHelmHooks) (eq .Values.controlPlane.environment "kubernetes"))) }} + {{ $hook := "pre-upgrade,pre-install" }} + {{- $serviceAccountName := printf "%s-install-crds" (include "kuma.name" .) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "{{ $hook }}" + "helm.sh/hook-weight": "-1" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +{{- with concat .Values.installCrdsOnUpgrade.imagePullSecrets .Values.global.imagePullSecrets | uniq }} +imagePullSecrets: + {{- range . }} + - name: {{ . | quote }} + {{- end }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "kuma.name" . }}-install-crds + annotations: + "helm.sh/hook": "{{ $hook }}" + "helm.sh/hook-weight": "-1" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +rules: + - apiGroups: + - "apiextensions.k8s.io" + resources: + - customresourcedefinitions + verbs: + - create + - patch + - update + - list + - get +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "kuma.name" . }}-install-crds + annotations: + "helm.sh/hook": "{{ $hook }}" + "helm.sh/hook-weight": "-1" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded,hook-failed" + labels: + {{- include "kuma.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "kuma.name" . }}-install-crds +subjects: + - kind: ServiceAccount + name: {{ $serviceAccountName }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "kuma.name" . }}-install-crds-scripts + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": "{{ $hook }}" + "helm.sh/hook-weight": "-1" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" + labels: + {{- include "kuma.labels" . | nindent 4 }} +data: + install_crds.sh: | + #!/usr/bin/env sh + set -e + + if [ -s /kuma/crds/crds.yaml ]; then + echo "/kuma/crds/crds.yaml found and is not empty, adding crds" + kubectl apply -f /kuma/crds/crds.yaml + else + echo "/kuma/crds/crds.yaml not found or empty, it looks like there is no crds to install" + fi + save_crds.sh: | + set -e + + crds="$(kumactl install crds --no-config)" + + if [ -n "${crds}" ]; then + echo "found crds - saving to /kuma/crds/crds.yaml" + echo "${crds}" > /kuma/crds/crds.yaml + fi +--- +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "kuma.name" . }}-install-crds + namespace: {{ .Release.Namespace }} + labels: + {{ include "kuma.labels" . | nindent 4 }} + annotations: + "helm.sh/hook": "{{ $hook }}" + "helm.sh/hook-delete-policy": "before-hook-creation,hook-succeeded" +spec: + template: + metadata: + name: {{ template "kuma.name" . }}-install-crds-job + labels: + {{ include "kuma.labels" . | nindent 8 }} + spec: + serviceAccountName: {{ $serviceAccountName }} + {{- with .Values.hooks.nodeSelector }} + nodeSelector: + {{ toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.hooks.tolerations }} + tolerations: + {{ toYaml . | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + securityContext: + {{- toYaml .Values.hooks.podSecurityContext | trim | nindent 8 }} + containers: + - name: pre-upgrade-job + image: "{{ .Values.kubectl.image.registry }}/{{ .Values.kubectl.image.repository }}:{{ .Values.kubectl.image.tag }}" + securityContext: + {{- toYaml (mergeOverwrite (dict "runAsUser" 65534) .Values.hooks.containerSecurityContext) | trim | nindent 12 }} + resources: + requests: + cpu: "100m" + memory: "256Mi" + limits: + cpu: "100m" + memory: "256Mi" + command: ["/kuma/scripts/install_crds.sh"] + volumeMounts: + - mountPath: /kuma/crds + name: crds + readOnly: true + - mountPath: /kuma/scripts + name: scripts + readOnly: true + initContainers: + - name: pre-upgrade-job-init + image: {{ include "kuma.formatImage" (dict "image" .Values.kumactl.image "root" $) | quote }} + securityContext: + {{- toYaml .Values.hooks.containerSecurityContext | trim | nindent 12 }} + resources: + requests: + cpu: "100m" + memory: "256Mi" + limits: + cpu: "100m" + memory: "256Mi" + volumeMounts: + - mountPath: /kuma/crds + name: crds + - mountPath: /kuma/scripts + name: scripts + readOnly: true + command: ["sh", "-c"] + args: ["/kuma/scripts/save_crds.sh"] + volumes: + - name: scripts + configMap: + name: {{ include "kuma.name" . }}-install-crds-scripts + defaultMode: 0755 + - name: crds + emptyDir: {} +{{- end }} diff --git a/charts/kuma/kuma/2.9.1/values.yaml b/charts/kuma/kuma/2.9.1/values.yaml new file mode 100644 index 000000000..766792e5b --- /dev/null +++ b/charts/kuma/kuma/2.9.1/values.yaml @@ -0,0 +1,903 @@ +global: + image: + # -- Default registry for all Kuma Images + registry: "docker.io/kumahq" + # -- The default tag for all Kuma images, which itself defaults to .Chart.AppVersion + tag: + # -- Add `imagePullSecrets` to all the service accounts used for Kuma components + imagePullSecrets: [] + +# -- Whether to patch the target namespace with the system label +patchSystemNamespace: true + +installCrdsOnUpgrade: + # -- Whether install new CRDs before upgrade (if any were introduced with the new version of Kuma) + enabled: true + # -- The `imagePullSecrets` to attach to the Service Account running CRD installation. + # This field will be deprecated in a future release, please use .global.imagePullSecrets + imagePullSecrets: [] + +# -- Whether to disable all helm hooks +noHelmHooks: false + +# -- Whether to restart control-plane by calculating a new checksum for the secret +restartOnSecretChange: true + +controlPlane: + # -- Environment that control plane is run in, useful when running universal global control plane on k8s + environment: "kubernetes" + + # -- Labels to add to resources in addition to default labels + extraLabels: {} + + # -- Kuma CP log level: one of off,info,debug + logLevel: "info" + + # -- Kuma CP log output path: Defaults to /dev/stdout + logOutputPath: "" + + # -- Kuma CP modes: one of zone,global + mode: "zone" + + # -- (string) Kuma CP zone, if running multizone + zone: + + # -- Only used in `zone` mode + kdsGlobalAddress: "" + + # -- Number of replicas of the Kuma CP. Ignored when autoscaling is enabled + replicas: 1 + + # -- Minimum number of seconds for which a newly created pod should be ready for it to be considered available. + minReadySeconds: 0 + + # -- Annotations applied only to the `Deployment` resource + deploymentAnnotations: {} + + # -- Annotations applied only to the `Pod` resource + podAnnotations: {} + + # Horizontal Pod Autoscaling configuration + autoscaling: + # -- Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster + enabled: false + + # -- The minimum CP pods to allow + minReplicas: 2 + # -- The max CP pods to scale to + maxReplicas: 5 + + # -- For clusters that don't support autoscaling/v2, autoscaling/v1 is used + targetCPUUtilizationPercentage: 80 + # -- For clusters that do support autoscaling/v2, use metrics + metrics: + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: 80 + + # -- Node selector for the Kuma Control Plane pods + nodeSelector: + kubernetes.io/os: linux + + # -- Tolerations for the Kuma Control Plane pods + tolerations: [] + + podDisruptionBudget: + # -- Whether to create a pod disruption budget + enabled: false + # -- The maximum number of unavailable pods allowed by the budget + maxUnavailable: 1 + + # -- Affinity placement rule for the Kuma Control Plane pods. + # This is rendered as a template, so you can reference other helm variables or includes. + affinity: + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + labelSelector: + # These match the selector labels used on the deployment. + matchExpressions: + - key: app.kubernetes.io/name + operator: In + values: + - '{{ include "kuma.name" . }}' + - key: app.kubernetes.io/instance + operator: In + values: + - '{{ .Release.Name }}' + - key: app + operator: In + values: + - '{{ include "kuma.name" . }}-control-plane' + topologyKey: kubernetes.io/hostname + + # -- Topology spread constraints rule for the Kuma Control Plane pods. + # This is rendered as a template, so you can use variables to generate match labels. + topologySpreadConstraints: + + # -- Failure policy of the mutating webhook implemented by the Kuma Injector component + injectorFailurePolicy: Fail + + service: + apiServer: + http: + # -- Port on which Http api server Service is exposed on Node for service of type NodePort + nodePort: 30681 + https: + # -- Port on which Https api server Service is exposed on Node for service of type NodePort + nodePort: 30682 + + # -- Whether to create a service resource. + enabled: true + + # -- (string) Optionally override of the Kuma Control Plane Service's name + name: + + # -- Service type of the Kuma Control Plane + type: ClusterIP + + # -- Annotations to put on the Kuma Control Plane + annotations: + prometheus.io/scrape: "true" + prometheus.io/port: "5680" + + # Kuma API and GUI ingress settings. Useful if you want to expose the + # API and GUI of Kuma outside the k8s cluster. + ingress: + # -- Install K8s Ingress resource that exposes GUI and API + enabled: false + # -- IngressClass defines which controller will implement the resource + ingressClassName: + # -- Ingress hostname + hostname: + # -- Map of ingress annotations. + annotations: {} + # -- Ingress path. + path: / + # -- Each path in an Ingress is required to have a corresponding path type. (ImplementationSpecific/Exact/Prefix) + pathType: ImplementationSpecific + # -- Port from kuma-cp to use to expose API and GUI. Switch to 5682 to expose TLS port + servicePort: 5681 + + globalZoneSyncService: + # -- Whether to create a k8s service for the global zone sync + # service. It will only be created when enabled and deploying the global + # control plane. + enabled: true + # -- Service type of the Global-zone sync + type: LoadBalancer + # -- (string) Optionally specify IP to be used by cloud provider when configuring load balancer + loadBalancerIP: + # -- Optionally specify allowed source ranges that can access the load balancer + loadBalancerSourceRanges: [] + # -- Additional annotations to put on the Global Zone Sync Service + annotations: { } + # -- Port on which Global Zone Sync Service is exposed on Node for service of type NodePort + nodePort: 30685 + # -- Port on which Global Zone Sync Service is exposed + port: 5685 + # -- Protocol of the Global Zone Sync service port + protocol: grpc + + defaults: + # -- Whether to skip creating the default Mesh + skipMeshCreation: false + + # -- Whether to automountServiceAccountToken for cp. Optionally set to false + automountServiceAccountToken: true + + # -- Optionally override the resource spec + resources: + requests: + cpu: 500m + memory: 256Mi + limits: + memory: 256Mi + + # -- Pod lifecycle settings (useful for adding a preStop hook, when + # using AWS ALB or NLB) + lifecycle: {} + + # -- Number of seconds to wait before force killing the pod. Make sure to + # update this if you add a preStop hook. + terminationGracePeriodSeconds: 30 + + # TLS for various servers + tls: + general: + # -- Secret that contains tls.crt, tls.key [and ca.crt when no + # controlPlane.tls.general.caSecretName specified] for protecting + # Kuma in-cluster communication + secretName: "" + # -- Secret that contains ca.crt that was used to sign cert for protecting + # Kuma in-cluster communication (ca.crt present in this secret + # have precedence over the one provided in the controlPlane.tls.general.secretName) + caSecretName: "" + # -- Base64 encoded CA certificate (the same as in controlPlane.tls.general.secret#ca.crt) + caBundle: "" + apiServer: + # -- Secret that contains tls.crt, tls.key for protecting Kuma API on HTTPS + secretName: "" + # -- Secret that contains list of .pem certificates that can access admin endpoints of Kuma API on HTTPS + clientCertsSecretName: "" + # - if not creating the global control plane, then do nothing + # - if secretName is empty and create is false, then do nothing + # - if secretName is non-empty and create is false, then use the secret made outside of helm with the name secretName + # - if secretName is empty and create is true, then create a secret with a default name and use it + # - if secretName is non-empty and create is true, then create the secret using the provided name + kdsGlobalServer: + # -- Name of the K8s TLS Secret resource. If you set this and don't set + # create=true, you have to create the secret manually. + secretName: "" + # -- Whether to create the TLS secret in helm. + create: false + # -- The TLS certificate to offer. + cert: "" + # -- The TLS key to use. + key: "" + # - if not creating the zonal control plane, then do nothing + # - if secretName is empty and create is false, then do nothing + # - if secretName is non-empty and create is false, then use the secret made outside of helm with the name secretName + # - if secretName is empty and create is true, then create a secret with a default name and use it + # - if secretName is non-empty and create is true, then create the secret using the provided name + kdsZoneClient: + # -- Name of the K8s Secret resource that contains ca.crt which was + # used to sign the certificate of KDS Global Server. If you set this + # and don't set create=true, you have to create the secret manually. + secretName: "" + # -- Whether to create the TLS secret in helm. + create: false + # -- CA bundle that was used to sign the certificate of KDS Global Server. + cert: "" + # -- If true, TLS cert of the server is not verified. + skipVerify: false + + # -- Annotations to add for Control Plane's Service Account + serviceAccountAnnotations: { } + + image: + # -- Kuma CP ImagePullPolicy + pullPolicy: IfNotPresent + # -- Kuma CP image repository + repository: "kuma-cp" + # -- Kuma CP Image tag. When not specified, the value is copied from global.tag + tag: + + # -- (object with { Env: string, Secret: string, Key: string }) Secrets to add as environment variables, + # where `Env` is the name of the env variable, + # `Secret` is the name of the Secret, + # and `Key` is the key of the Secret value to use + secrets: + # someSecret: + # Secret: some-secret + # Key: secret_key + # Env: SOME_SECRET + + # -- Additional environment variables that will be passed to the control plane + envVars: { } + + # -- Additional environment variables that will be passed to the control plane. Can be used with Kubernetes downward API + envVarEntries: + # - name: MY_NODE_NAME + # valueFrom: + # fieldRef: + # fieldPath: spec.nodeName + + # -- Additional config maps to mount into the control plane, with optional inline values + extraConfigMaps: [ ] +# - name: extra-config +# mountPath: /etc/extra-config +# readOnly: true +# values: +# extra-config-key: | +# extra-config-value + + # -- (object with { name: string, mountPath: string, readOnly: string }) Additional secrets to mount into the control plane, + # where `Env` is the name of the env variable, + # `Secret` is the name of the Secret, + # and `Key` is the key of the Secret value to use + extraSecrets: + # extraConfig: + # name: extra-config + # mountPath: /etc/extra-config + # readOnly: true + + webhooks: + validator: + # -- Additional rules to apply on Kuma validator webhook. Useful when building custom policy on top of Kuma. + additionalRules: "" + ownerReference: + # -- Additional rules to apply on Kuma owner reference webhook. Useful when building custom policy on top of Kuma. + additionalRules: "" + + # -- Specifies if the deployment should be started in hostNetwork mode. + hostNetwork: false + # -- Define a new server port for the admission controller. Recommended to set in combination with + # hostNetwork to prevent multiple port bindings on the same port (like Calico in AWS EKS). + admissionServerPort: 5443 + + # -- Security context at the pod level for control plane. + podSecurityContext: + runAsNonRoot: true + + # -- Security context at the container level for control plane. + containerSecurityContext: + readOnlyRootFilesystem: true + + # -- If true, then control plane can support TLS secrets for builtin gateway outside of mesh system namespace. + # The downside is that control plane requires permission to read Secrets in all namespaces. + supportGatewaySecretsInAllNamespaces: false + # -- DNS configuration for the control-plane pod. + # This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). + dns: + # -- Defines how DNS resolution is configured for that Pod. + policy: "" + # -- Optional dns configuration, required when policy is 'None' + config: + # -- A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. + nameservers: [] + # -- A list of DNS search domains for hostname lookup in the Pod. + searches: [] + +cni: + # -- Install Kuma with CNI instead of proxy init container + enabled: false + # -- Install CNI in chained mode + chained: false + # -- Set the CNI install directory + netDir: /etc/cni/multus/net.d + # -- Set the CNI bin directory + binDir: /var/lib/cni/bin + # -- Set the CNI configuration name + confName: kuma-cni.conf + # -- CNI log level: one of off,info,debug + logLevel: info + # -- Node Selector for the CNI pods + nodeSelector: + kubernetes.io/os: linux + # -- Tolerations for the CNI pods + tolerations: [] + # -- Additional pod annotations + podAnnotations: { } + # -- Set the CNI namespace + namespace: kube-system + + image: + # -- CNI image repository + repository: "kuma-cni" + # -- CNI image tag - defaults to .Chart.AppVersion + tag: + # -- CNI image pull policy + imagePullPolicy: IfNotPresent + + # -- it's only useful in tests to trigger a possible race condition + delayStartupSeconds: 0 + + # -- use new CNI (experimental) + experimental: + imageEbpf: + # -- CNI experimental eBPF image registry + registry: "docker.io/kumahq" + # -- CNI experimental eBPF image repository + repository: "merbridge" + # -- CNI experimental eBPF image tag + tag: "0.8.5" + + resources: + requests: + cpu: 100m + memory: 100Mi + limits: + memory: 100Mi + + # -- Security context at the pod level for cni + podSecurityContext: {} + + # -- Security context at the container level for cni + containerSecurityContext: + readOnlyRootFilesystem: true + runAsNonRoot: false + runAsUser: 0 + runAsGroup: 0 + +dataPlane: + # -- If true, then turn on CoreDNS query logging + dnsLogging: false + image: + # -- The Kuma DP image repository + repository: "kuma-dp" + # -- Kuma DP ImagePullPolicy + pullPolicy: IfNotPresent + # -- Kuma DP Image Tag. When not specified, the value is copied from global.tag + tag: + + initImage: + # -- The Kuma DP init image repository + repository: "kuma-init" + # -- Kuma DP init image tag When not specified, the value is copied from global.tag + tag: + +ingress: + # -- If true, it deploys Ingress for cross cluster communication + enabled: false + + # -- Labels to add to resources, in addition to default labels + extraLabels: {} + + # -- Time for which old listener will still be active as draining + drainTime: 30s + + # -- Number of replicas of the Ingress. Ignored when autoscaling is enabled. + replicas: 1 + + # -- Log level for ingress (available values: off|info|debug) + logLevel: info + + # -- Define the resources to allocate to mesh ingress + resources: + requests: + cpu: 50m + memory: 64Mi + limits: + cpu: 1000m + memory: 512Mi + + # -- Pod lifecycle settings (useful for adding a preStop hook, when + # using AWS ALB or NLB) + lifecycle: {} + + # -- Number of seconds to wait before force killing the pod. Make sure to + # update this if you add a preStop hook. + terminationGracePeriodSeconds: 40 + + # Horizontal Pod Autoscaling configuration + autoscaling: + # -- Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster + enabled: false + + # -- The minimum CP pods to allow + minReplicas: 2 + # -- The max CP pods to scale to + maxReplicas: 5 + + # -- For clusters that don't support autoscaling/v2, autoscaling/v1 is used + targetCPUUtilizationPercentage: 80 + # -- For clusters that do support autoscaling/v2, use metrics + metrics: + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: 80 + + service: + # -- Whether to create a Service resource. + enabled: true + # -- Service type of the Ingress + type: LoadBalancer + # -- (string) Optionally specify IP to be used by cloud provider when configuring load balancer + loadBalancerIP: + # -- Additional annotations to put on the Ingress service + annotations: { } + # -- Port on which Ingress is exposed + port: 10001 + # -- Port on which service is exposed on Node for service of type NodePort + nodePort: + # -- Additional pod annotations (deprecated favor `podAnnotations`) + annotations: { } + # -- Additional pod annotations + podAnnotations: { } + # -- Node Selector for the Ingress pods + nodeSelector: + kubernetes.io/os: linux + # -- Tolerations for the Ingress pods + tolerations: [] + podDisruptionBudget: + # -- Whether to create a pod disruption budget + enabled: false + # -- The maximum number of unavailable pods allowed by the budget + maxUnavailable: 1 + + # -- Affinity placement rule for the Kuma Ingress pods + # This is rendered as a template, so you can reference other helm variables + # or includes. + affinity: + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + labelSelector: + # These match the selector labels used on the deployment. + matchExpressions: + - key: app.kubernetes.io/name + operator: In + values: + - '{{ include "kuma.name" . }}' + - key: app.kubernetes.io/instance + operator: In + values: + - '{{ .Release.Name }}' + - key: app + operator: In + values: + - kuma-ingress + topologyKey: kubernetes.io/hostname + + # -- Topology spread constraints rule for the Kuma Mesh Ingress pods. + # This is rendered as a template, so you can use variables to generate match labels. + topologySpreadConstraints: + + # -- Security context at the pod level for ingress + podSecurityContext: + runAsNonRoot: true + runAsUser: 5678 + runAsGroup: 5678 + + # -- Security context at the container level for ingress + containerSecurityContext: + readOnlyRootFilesystem: true + + # -- Annotations to add for Control Plane's Service Account + serviceAccountAnnotations: { } + # -- Whether to automountServiceAccountToken for cp. Optionally set to false + automountServiceAccountToken: true + # -- DNS configuration for the ingress pod. + # This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). + dns: + # -- Defines how DNS resolution is configured for that Pod. + policy: "" + # -- Optional dns configuration, required when policy is 'None' + config: + # -- A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. + nameservers: [] + # -- A list of DNS search domains for hostname lookup in the Pod. + searches: [] + +egress: + # -- If true, it deploys Egress for cross cluster communication + enabled: false + # -- Labels to add to resources, in addition to the default labels. + extraLabels: {} + # -- Time for which old listener will still be active as draining + drainTime: 30s + # -- Number of replicas of the Egress. Ignored when autoscaling is enabled. + replicas: 1 + + # -- Log level for egress (available values: off|info|debug) + logLevel: info + + # Horizontal Pod Autoscaling configuration + autoscaling: + # -- Whether to enable Horizontal Pod Autoscaling, which requires the [Metrics Server](https://github.com/kubernetes-sigs/metrics-server) in the cluster + enabled: false + + # -- The minimum CP pods to allow + minReplicas: 2 + # -- The max CP pods to scale to + maxReplicas: 5 + + # -- For clusters that don't support autoscaling/v2, autoscaling/v1 is used + targetCPUUtilizationPercentage: 80 + # -- For clusters that do support autoscaling/v2, use metrics + metrics: + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: 80 + resources: + requests: + cpu: 50m + memory: 64Mi + limits: + cpu: 1000m + memory: 512Mi + + service: + # -- Whether to create the service object + enabled: true + # -- Service type of the Egress + type: ClusterIP + # -- (string) Optionally specify IP to be used by cloud provider when configuring load balancer + loadBalancerIP: + # -- Additional annotations to put on the Egress service + annotations: { } + # -- Port on which Egress is exposed + port: 10002 + # -- Port on which service is exposed on Node for service of type NodePort + nodePort: + # -- Additional pod annotations (deprecated favor `podAnnotations`) + annotations: { } + # -- Additional pod annotations + podAnnotations: { } + # -- Node Selector for the Egress pods + nodeSelector: + kubernetes.io/os: linux + # -- Tolerations for the Egress pods + tolerations: [] + podDisruptionBudget: + # -- Whether to create a pod disruption budget + enabled: false + # -- The maximum number of unavailable pods allowed by the budget + maxUnavailable: 1 + + # -- Affinity placement rule for the Kuma Egress pods. + # This is rendered as a template, so you can reference other helm variables or includes. + affinity: + podAntiAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 100 + podAffinityTerm: + labelSelector: + # These match the selector labels used on the deployment. + matchExpressions: + - key: app.kubernetes.io/name + operator: In + values: + - '{{ include "kuma.name" . }}' + - key: app.kubernetes.io/instance + operator: In + values: + - '{{ .Release.Name }}' + - key: app + operator: In + values: + - kuma-egress + topologyKey: kubernetes.io/hostname + + # -- Topology spread constraints rule for the Kuma Egress pods. + # This is rendered as a template, so you can use variables to generate match labels. + topologySpreadConstraints: + + # -- Security context at the pod level for egress + podSecurityContext: + runAsNonRoot: true + runAsUser: 5678 + runAsGroup: 5678 + + # -- Security context at the container level for egress + containerSecurityContext: + readOnlyRootFilesystem: true + + # -- Annotations to add for Control Plane's Service Account + serviceAccountAnnotations: { } + # -- Whether to automountServiceAccountToken for cp. Optionally set to false + automountServiceAccountToken: true + # -- DNS configuration for the egress pod. + # This is equivalent to the [Kubernetes DNS policy](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-s-dns-policy). + dns: + # -- Defines how DNS resolution is configured for that Pod. + policy: "" + # -- Optional dns configuration, required when policy is 'None' + config: + # -- A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified. + nameservers: [] + # -- A list of DNS search domains for hostname lookup in the Pod. + searches: [] + +kumactl: + image: + # -- The kumactl image repository + repository: kumactl + # -- The kumactl image tag. When not specified, the value is copied from global.tag + tag: + +kubectl: + image: + # -- The kubectl image registry + registry: docker.io + # -- The kubectl image repository + repository: bitnami/kubectl + # -- The kubectl image tag + tag: "1.27.5" +hooks: + # -- Node selector for the HELM hooks + nodeSelector: + kubernetes.io/os: linux + # -- Tolerations for the HELM hooks + tolerations: [] + # -- Security context at the pod level for crd/webhook/ns + podSecurityContext: + runAsNonRoot: true + + # -- Security context at the container level for crd/webhook/ns + containerSecurityContext: + readOnlyRootFilesystem: true + + # -- ebpf-cleanup hook needs write access to the root filesystem to clean ebpf programs + # Changing below values will potentially break ebpf cleanup completely, + # so be cautious when doing so. + ebpfCleanup: + # -- Security context at the pod level for crd/webhook/cleanup-ebpf + podSecurityContext: + runAsNonRoot: false + # -- Security context at the container level for crd/webhook/cleanup-ebpf + containerSecurityContext: + readOnlyRootFilesystem: false + +transparentProxy: + configMap: + # -- If true, enables the use of a ConfigMap to manage transparent proxy configuration + # instead of directly configuring it within the Kuma system + enabled: false + # -- The name of the ConfigMap used to store the transparent proxy configuration + name: kuma-transparent-proxy-config + config: + # -- The username or UID of the user that will run kuma-dp. If not provided, the system will + # use the default UID ("5678") or the default username ("kuma-dp") + kumaDPUser: "5678" + # -- The IP family mode used for configuring traffic redirection in the transparent proxy + # Supports "dualstack" (for both IPv4 and IPv6) and "ipv4" modes + ipFamilyMode: dualstack + redirect: + dns: + # -- Enables DNS redirection in the transparent proxy + enabled: true + # -- Redirect all DNS queries + captureAll: true + # -- The port on which the DNS server listens + port: 15053 + # -- Path to the system's resolv.conf file + resolvConfigPath: /etc/resolv.conf + # -- Disables conntrack zone splitting, which can prevent potential DNS issues + skipConntrackZoneSplit: false + inbound: + # -- Enables inbound traffic redirection + enabled: true + # -- Port used for redirecting inbound traffic + port: 15006 + # -- List of ports to exclude from inbound traffic redirection + excludePorts: [] + # -- List of IP addresses to exclude from inbound traffic redirection for specific ports + excludePortsForIPs: [] + # -- List of UIDs to exclude from inbound traffic redirection for specific ports + excludePortsForUIDs: [] + # -- List of ports to include in inbound traffic redirection + includePorts: [] + # -- Inserts the redirection rule at the beginning of the chain instead of appending it + insertRedirectInsteadOfAppend: false + outbound: + # -- Enables outbound traffic redirection + enabled: true + # -- Port used for redirecting outbound traffic + port: 15001 + # -- List of ports to exclude from outbound traffic redirection + excludePorts: [] + # -- List of IP addresses to exclude from outbound traffic redirection for specific ports + excludePortsForIPs: [] + # -- List of UIDs to exclude from outbound traffic redirection for specific ports + excludePortsForUIDs: [] + # -- List of ports to include in outbound traffic redirection + includePorts: [] + # -- Inserts the redirection rule at the beginning of the chain instead of appending it + insertRedirectInsteadOfAppend: false + vnet: + # -- Specifies virtual networks using the format interfaceName:CIDR + # Allows matching traffic on specific network interfaces + # Examples: + # - "docker0:172.17.0.0/16" + # - "br+:172.18.0.0/16" (matches any interface starting with "br") + # - "iface:::1/64" (for IPv6) + networks: [] + ebpf: + # -- Enables eBPF support for handling traffic redirection in the transparent proxy + enabled: false + # -- The path of the BPF filesystem + bpffsPath: /run/kuma/bpf + # -- The path of cgroup2 + cgroupPath: /sys/fs/cgroup + # -- The name of the environment variable containing the IP address of the instance (pod/vm) + # where transparent proxy will be installed + instanceIPEnvVarName: "" + # -- Path where compiled eBPF programs and other necessary files for eBPF mode can be found + programsSourcePath: /tmp/kuma-ebpf + # -- The network interface for TC eBPF programs to bind to. If not provided, it will be + # automatically determined + tcAttachIface: "" + retry: + # -- The maximum number of retry attempts for operations + maxRetries: 4 + # -- The time duration to wait between retry attempts + sleepBetweenRetries: 2s + iptablesExecutables: + # -- Custom path for the iptables executable (IPv4) + iptables: "" + # -- Custom path for the iptables-save executable (IPv4) + iptables-save: "" + # -- Custom path for the iptables-restore executable (IPv4) + iptables-restore: "" + # -- Custom path for the ip6tables executable (IPv6) + ip6tables: "" + # -- Custom path for the ip6tables-save executable (IPv6) + ip6tables-save: "" + # -- Custom path for the ip6tables-restore executable (IPv6) + ip6tables-restore: "" + log: + # -- Enables logging of iptables rules for diagnostics and monitoring + enabled: false + comments: + # -- Disables comments in the generated iptables rules + disabled: false + # -- Time in seconds to wait for acquiring the xtables lock before failing + # Value 0 means wait indefinitely + wait: 5 + # -- Time interval between retries to acquire the xtables lock in seconds + waitInterval: 0 + # -- Drops invalid packets to avoid connection resets in high-throughput scenarios + dropInvalidPackets: false + # -- Enables firewalld support to store iptables rules + storeFirewalld: false + # -- Enables verbose mode with longer argument/flag names and additional comments + verbose: false + +experimental: + # Configuration for the experimental ebpf mode for transparent proxy + ebpf: + # -- If true, ebpf will be used instead of using iptables to install/configure transparent proxy + enabled: false + # -- Name of the environmental variable which will contain the IP address of a pod + instanceIPEnvVarName: INSTANCE_IP + # -- Path where BPF file system should be mounted + bpffsPath: /sys/fs/bpf + # -- Host's cgroup2 path + cgroupPath: /sys/fs/cgroup + # -- Name of the network interface which TC programs should be attached to, we'll try to automatically determine it if empty + tcAttachIface: "" + # -- Path where compiled eBPF programs which will be installed can be found + programsSourcePath: /tmp/kuma-ebpf + # -- If true, enable native Kubernetes sidecars. This requires at least + # Kubernetes v1.29 + sidecarContainers: false + +# Postgres' settings for universal control plane on k8s +postgres: + # -- Postgres port, password should be provided as a secret reference in "controlPlane.secrets" + # with the Env value "KUMA_STORE_POSTGRES_PASSWORD". + # Example: + # controlPlane: + # secrets: + # - Secret: postgres-postgresql + # Key: postgresql-password + # Env: KUMA_STORE_POSTGRES_PASSWORD + port: "5432" + # TLS settings + tls: + # -- Mode of TLS connection. Available values are: "disable", "verifyNone", "verifyCa", "verifyFull" + mode: disable # ENV: KUMA_STORE_POSTGRES_TLS_MODE + # -- Whether to disable SNI the postgres `sslsni` option. + disableSSLSNI: false # ENV: KUMA_STORE_POSTGRES_TLS_DISABLE_SSLSNI + # -- Secret name that contains the ca.crt + caSecretName: + # -- Secret name that contains the client tls.crt, tls.key + secretName: + +# @ignored for helm-docs +plugins: + resources: + hostnamegenerators: true + meshexternalservices: true + meshmultizoneservices: true + meshservices: true + policies: + meshaccesslogs: true + meshcircuitbreakers: true + meshfaultinjections: true + meshhealthchecks: true + meshhttproutes: true + meshloadbalancingstrategies: true + meshmetrics: true + meshpassthroughs: true + meshproxypatches: true + meshratelimits: true + meshretries: true + meshtcproutes: true + meshtimeouts: true + meshtlses: true + meshtraces: true + meshtrafficpermissions: true diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/Chart.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/Chart.yaml new file mode 100644 index 000000000..341bdd9e5 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/Chart.yaml @@ -0,0 +1,21 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: NetScaler CPX with Ingress Controller + catalog.cattle.io/kube-version: '>=v1.16.0-0' + catalog.cattle.io/release-name: netscaler-cpx-with-ingress-controller +apiVersion: v2 +appVersion: 2.2.10 +description: A Helm chart for NetScaler CPX with NetScaler ingress Controller running + as sidecar. +home: https://www.netscaler.com +icon: file://assets/icons/netscaler-cpx-with-ingress-controller.png +kubeVersion: '>=v1.16.0-0' +maintainers: +- email: priyanka.sharma@cloud.com + name: priyankash-citrix +- email: subash.dangol@cloud.com + name: subashd +name: netscaler-cpx-with-ingress-controller +sources: +- https://github.com/netscaler/netscaler-k8s-ingress-controller +version: 2.2.10 diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/README.md b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/README.md new file mode 100644 index 000000000..c3dca3ff0 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/README.md @@ -0,0 +1,747 @@ +# NetScaler CPX with NetScaler Ingress Controller running as sidecar. + +In a [Kubernetes](https://kubernetes.io/) or [OpenShift](https://www.openshift.com) cluster, you can deploy [NetScaler CPX](https://docs.netscaler.com/en-us/citrix-adc-cpx/cpx/) with NetScaler ingress controller as a [sidecar](https://kubernetes.io/docs/concepts/workloads/pods/pod-overview/). The NetScaler CPX instance is used for load balancing the North-South traffic to the microservices in your cluster. And, the sidecar NetScaler ingress controller configures the NetScaler CPX. + +## TL;DR; + +### For Kubernetes + ``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + + helm install netscaler-cpx-with-ingress-controller netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes + ``` + +### For OpenShift + + ``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + + helm install netscaler-cpx-with-ingress-controller netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,openshift=true + ``` + +> **Important:** +> +> The "license.accept" is a mandatory argument and should be set to "yes" to accept the terms of the NetScaler license. + +> **NOTE:** +> +> The CRDs supported by NetScaler will be installed automatically with the installation of the Helm Charts if CRDs are not already available in the cluster. + +## Introduction +This Helm chart deploys a NetScaler CPX with NetScaler ingress controller as a sidecar in the [Kubernetes](https://kubernetes.io/) or in the [Openshift](https://www.openshift.com) cluster using the [Helm](https://helm.sh/) package manager. + +### Prerequisites + +- The [Kubernetes](https://kubernetes.io/) version should be 1.16 and above if using Kubernetes environment. +- The [Openshift](https://www.openshift.com) version 4.8 or later if using OpenShift platform. +- The [Helm](https://helm.sh/) version 3.x or later. You can follow instruction given [here](https://github.com/netscaler/netscaler-helm-charts/blob/master/Helm_Installation_version_3.md) to install the same. +- You have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator), if you want to view the metrics of the NetScaler CPX collected by the [metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics). +- Registration of NetScaler CPX in ADM: You may want to register your CPX in ADM for licensing or to obtain [servicegraph](https://docs.netscaler.com/en-us/citrix-application-delivery-management-service/application-analytics-and-management/service-graph.html). For this you will have to create a Kubernetes secret using ADM credentials and provide it while install the chart. Create a Kubernetes secret for the user name and password using the following command: + + ``` + kubectl create secret generic admlogin --from-literal=username= --from-literal=password= + ``` + +## Installing the Chart +Add the NetScaler Ingress Controller helm chart repository using command: + + ``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + ``` + +### For Kubernetes: +#### 1. NetScaler CPX with NetScaler Ingress Controller running as side car. +To install the chart with the release name ``` my-release```: + + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,ingressClass[0]= + ``` + +> **Note:** +> +> By default the chart installs the recommended [RBAC](https://kubernetes.io/docs/admin/authorization/rbac/) roles and role bindings. + +The command deploys NetScaler CPX with NetScaler ingress controller as a sidecar on the Kubernetes cluster with the default configuration. The [configuration](#configuration) section lists the mandatory and optional parameters that you can configure during installation. + +#### 2. NetScaler CPX with NetScaler Ingress Controller and Exporter running as side car. +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed as sidecar to the NetScaler CPX and collects metrics from the NetScaler CPX instance. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. +> **Note:** +> +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +Use the following command for this: + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,ingressClass[0]=,exporter.required=true + ``` + +### For OpenShift: +Add the name of the service account created when the chart is deployed to the privileged Security Context Constraints of OpenShift: + + ``` + oc adm policy add-scc-to-user privileged system:serviceaccount:: + ``` + +#### 1. NetScaler CPX with NetScaler Ingress Controller running as side car. +To install the chart with the release name, `my-release`, use the following command: + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,openshift=true + ``` + +#### 2. NetScaler CPX with NetScaler Ingress Controller and Exporter running as side car. +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed as sidecar to the NetScaler CPX and collects metrics from the NetScaler CPX instance. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. +> **Note:** +> +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +Use the following command for this: + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,openshift=true,exporter.required=true + ``` + +### Installed components + +The following components are installed: + +- [NetScaler CPX](https://docs.netscaler.com/en-us/citrix-adc-cpx/current-release/about) +- [NetScaler ingress controller](https://github.com/netscaler/netscaler-k8s-ingress-controller) (if enabled) +- [Exporter](https://github.com/netscaler/netscaler-adc-metrics-exporter) (if enabled) + + +### NetScaler CPX Service Annotations: + + The parameter `serviceAnnotations` can be used to annotate CPX service while installing NetScaler CPX using this helm chart. + For example, if CPX is getting deployed in Azure and an Azure Internal Load Balancer is required before CPX then the annotation `service.beta.kubernetes.io/azure-load-balancer-internal:True` can be set in CPX service using Helm command: + + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,serviceAnnotations.service\\.beta\\.kubernetes\\.io/azure-load-balancer-internal=True + ``` + + or the same can be provided in [values.yaml](https://github.com/netscaler/netscaler-helm-charts/blob/master/netscaler-cpx-with-ingress-controller/values.yaml): + + ``` + license: + accept: yes + serviceAnnotations: + service.beta.kubernetes.io/azure-load-balancer-internal: True + ``` + + which can be used to install NetScaler CPX using Helm command: + + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller -f values.yaml + ``` + + To know more about service annotations supported by Kubernetes on various platforms please see [this](https://kubernetes.io/docs/concepts/services-networking/service/). + +### NetScaler CPX Service Ports: + + By default, port 80 and 443 of CPX service will exposed when CPX is installed using this helm chart. If it is required to expose any other ports in CPX service then the parameter `servicePorts` can be used for it. + For example, if port 9999 is required to be exposed then below helm command can be used for installing NetScaler CPX: + + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,servicePorts[0].port=9999,servicePorts[0].protocol=TCP,servicePorts[0].name=https + ``` + + or the same can be provided in [values.yaml](https://github.com/netscaler/netscaler-helm-charts/blob/master/netscaler-cpx-with-ingress-controller/values.yaml): + + ``` + license: + accept: yes + servicePorts: + - port: 9090 + protocol: TCP + name: https + ``` + + which can be used to install NetScaler using Helm command: + + ``` + helm install my-release netscaler/netscaler-cpx-with-ingress-controller -f values.yaml + ``` + +> **Note:** If `servicePorts` parameters is used, only ports provided in this parameter will be exposed in CPX service. +> If you want to expose default ports 80 or 443, then you will need to explicity mention these also in this parameter. + +### Configuration for ServiceGraph: + If NetScaler CPX need to send data to the NetScaler ADM to bring up the servicegraph, then the below steps can be followed to install NetScaler CPX with ingress controller. NetScaler ingress controller configures NetScaler CPX with the configuration required for servicegraph. + + 1. Create secret using NetScaler Agent credentials, which will be used by NetScaler CPX to communicate with NetScaler ADM Agent: + + kubectl create secret generic admlogin --from-literal=username= --from-literal=password= + + 2. Deploy NetScaler CPX with NetScaler ingress controller using helm command: + + helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,analyticsConfig.required=true,analyticsConfig.timeseries.metrics.enable=true,analyticsConfig.distributedTracing.enable=true,analyticsConfig.endpoint.metrics.service=,ADMSettings.ADMIP=,ADMSettings.loginSecret= + +> **Note:** +> If container agent is being used here for NetScaler ADM, please provide `svcIP` of container agent in the `analyticsConfig.endpoint.metrics.service` parameter. + +## NetScaler CPX DaemonSet with NetScaler Ingress Controller as sidecar for BGP Advertisement + + The previous section of deploying CPX as a Deployment requires a Tier-1 Loadbalancer such as NetScaler VPX or cloud loadbalancers to route the traffic to CPX instances running in Kubernetes cluster, but you can also leverage BGP network fabric in your on-prem environemnt to route the traffic to CPX instances in a Kubernetes or Openshift cluster. you need to deploy CPX with NetScaler Ingress Controller as Daemonset to advertise the ExternalIPs of the K8s services of type LoadBalancer to your BGP Fabric. NetScaler CPX establishes a BGP peering session with your network routers, and uses that peering session to advertise the IP addresses of external cluster services. If your routers have ECMP capability, the traffic is load-balanced to multiple CPX instances by the upstream router, which in turn load-balances to actual application pods. When you deploy the NetScaler CPX with this mode, NetScaler CPX adds iptables rules for each service of type LoadBalancer on Kubernetes nodes. The traffic destined to the external IP address is routed to NetScaler CPX pods. You can also set the 'ingressIP' variable to an IP Address to advertise the External IP address for Ingress resources. Refer [documentation](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/bgp-enhancement.md) for complete details about BGP advertisement with CPX. + +### Download the chart +You can download the chart usimg `helm pull` command. +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ +helm pull netscaler/netscaler-cpx-with-ingress-controller +tar -zxvf netscaler-cpx-with-ingress-controller-x.y.z.tgz +``` + +### Edit the BGP configuration in values.yaml +BGP configurations enables CPX to peer with neighbor routers for advertisting the routes for Service of Type LoadBalancer. NetScaler Ingress Controllers uses static IPs given in Service YAML or using an IPAM controller to allocate an External IP address, and same is advertisted to the neighbour router with the Gateway as Node IP. An example BGP configurations is given below. + +``` +# BGP configurations: local AS, remote AS and remote address is mandatory to provide. +bgpSettings: + required: true + bgpConfig: + - bgpRouter: + # Local AS number for BGP advertisement + localAS: + neighbor: + # Address of the nighbor router for BGP advertisement + - address: xx.xx.xx.xx + # Remote AS number + remoteAS: + advertisementInterval: 10 + ASOriginationInterval: 10 +``` +If the cluster spawns across multiple networks, you can also specify the NodeSelector to give different neighbors for different Cluster Nodes as shown below. + +``` +bgpSettings: + required: true + bgpConfig: + - nodeSelector: datacenter=ds1 + bgpRouter: + localAS: + neighbor: + - address: xx.xx.xx.xx + remoteAS: + advertisementInterval: 10 + ASOriginationInterval: 10 + - nodeSelector: datacenter=ds2 + bgpRouter: + localAS: + neighbor: + - address: yy.yy.yy.yy + remoteAS: + advertisementInterval: 10 + ASOriginationInterval: 10 +``` + +### Deploy the chart +#### For Kubernetes: +#### 1. NetScaler CPX DaemonSet with NetScaler Ingress Controller running as side car for BGP Advertisement. + + +To install the chart with the release name ``` my-release```: + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true + ``` +If you are running NetScaler IPAM for auto allocation of IPs for Service of type LoadBalancer, you must enable the IPAM configurations in NetScaler Ingress Controller as show below: + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ipam=true + ``` +If you are using ingress resources, you must set the `ingressIP` to a valid IP Address which will enable the BGP route advertisement for this IP when ingress resource is deployed. + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ingressIP= + ``` + +> **Note:** +> +> By default the chart installs the recommended [RBAC](https://kubernetes.io/docs/admin/authorization/rbac/) roles and role bindings. + +The command deploys NetScaler CPX Daemonset with NetScaler ingress controller as a sidecar on the Kubernetes cluster with the default configuration. The [configuration](#configuration) section lists the mandatory and optional parameters that you can configure during installation. + +#### 2. NetScaler CPX with NetScaler Ingress Controller and Exporter running as side car for BGP Advertisement. +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed as sidecar to the NetScaler CPX and collects metrics from the NetScaler CPX instance. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. +> **Note:** +> +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +Use the following command for this: + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,exporter.required=true + ``` +If you are running NetScaler IPAM controller for auto allocation of IPs for Service of type LoadBalancer, you must enable the IPAM configurations in NetScaler Ingress Controller as show below: + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ipam=true,exporter.required=true + ``` +If you are using ingress resources, you must set the `ingressIP` to a valid IP Address which will enable the BGP route advertisement for this IP when ingress resource is deployed. + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ingressIP=, exporter.required=true + ``` + +#### For OpenShift: +Add the name of the service account created when the chart is deployed to the privileged Security Context Constraints of OpenShift: + + ``` + oc adm policy add-scc-to-user privileged system:serviceaccount:: + ``` + +#### 1. NetScaler CPX DaemonSet with NetScaler Ingress Controller running as side car for BGP Advertisement. +To install the chart with the release name, `my-release`, use the following command: + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,openshift=true + ``` +If you are running NetScaler IPAM controller for auto allocation of IPs for Service of type LoadBalancer, you must enable the IPAM configurations in NetScaler Ingress Controller as show below: + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ipam=true,openshift=true + ``` + + If you are using ingress or Route resources, you must set the `ingressIP` to a valid IP Address which will enable the BGP route advertisement for this IP when ingress resource is deployed. + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ingressIP=,openshift=true + ``` + +#### 2. NetScaler CPX with NetScaler Ingress Controller and Exporter running as side car for BGP Advertisement. +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed as sidecar to the NetScaler CPX and collects metrics from the NetScaler CPX instance. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. +> **Note:** +> +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +Use the following command for this: + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,openshift=true,openshift=true,exporter.required=true + ``` +If you are running NetScaler IPAM controller for auto allocation of IPs for Service of type LoadBalancer, you must enable the IPAM configurations in NetScaler Ingress Controller as show below: + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ipam=true,openshift=true,exporter.required=true + ``` + +If you are using ingress or Route resources, you must set the `ingressIP` to a valid IP Address which will enable the BGP route advertisement for this IP when ingress resource is deployed. + + ``` + helm install my-release ./netscaler-cpx-with-ingress-controller --set license.accept=yes,cpxBgpRouter=true,ingressIP=,openshift=true,exporter.required=true + ``` + +## CRDs configuration + +CRDs will be installed when we install NetScaler ingress controller via Helm automatically if CRDs are not installed in cluster already. If you wish to skip the CRD installation step, you can pass the --skip-crds flag. For more information about this option in Helm please see [this](https://helm.sh/docs/chart_best_practices/custom_resource_definitions/). + +There are a few examples of how to use these CRDs, which are placed in the folder: [Example-CRDs](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds). Refer to them and install as needed, using the following command: +```kubectl create -f ``` + +### Details of the supported CRDs: + +#### authpolicies CRD: + +Authentication policies are used to enforce access restrictions to resources hosted by an application or an API server. + +NetScaler provides a Kubernetes CustomResourceDefinitions (CRDs) called the [Auth CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/auth) that you can use with the NetScaler ingress controller to define authentication policies on the ingress NetScaler. + +Example file: [auth_example.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/auth_example.yaml) + +#### continuousdeployments CRD for canary: + +Canary release is a technique to reduce the risk of introducing a new software version in production by first rolling out the change to a small subset of users. After user validation, the application is rolled out to the larger set of users. NetScaler-Integrated [Canary Deployment solution](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/canary) stitches together all components of continuous delivery (CD) and makes canary deployment easier for the application developers. + +#### httproutes and listeners CRDs for contentrouting: + +[Content Routing (CR)](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/contentrouting) is the execution of defined rules that determine the placement and configuration of network traffic between users and web applications, based on the content being sent. For example, a pattern in the URL or header fields of the request. + +Example files: [HTTPRoute_crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/HTTPRoute_crd.yaml), [Listener_crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/Listener_crd.yaml) + +#### ratelimits CRD: + +In a Kubernetes deployment, you can [rate limit the requests](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/ratelimit) to the resources on the back end server or services using rate limiting feature provided by the ingress NetScaler. + +Example files: [ratelimit-example1.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/ratelimit-example1.yaml), [ratelimit-example2.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/ratelimit-example2.yaml) + +#### vips CRD: + +NetScaler provides a CustomResourceDefinitions (CRD) called [VIP](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/vip) for asynchronous communication between the IPAM controller and NetScaler ingress controller. + +The IPAM controller is provided by NetScaler for IP address management. It allocates IP address to the service from a defined IP address range. The NetScaler ingress controller configures the IP address allocated to the service as virtual IP (VIP) in NetScaler VPX. And, the service is exposed using the IP address. + +When a new service is created, the NetScaler ingress controller creates a CRD object for the service with an empty IP address field. The IPAM Controller listens to addition, deletion, or modification of the CRD and updates it with an IP address to the CRD. Once the CRD object is updated, the NetScaler ingress controller automatically configures NetScaler-specfic configuration in the tier-1 NetScaler VPX. + +#### rewritepolicies CRD: + +In kubernetes environment, to deploy specific layer 7 policies to handle scenarios such as, redirecting HTTP traffic to a specific URL, blocking a set of IP addresses to mitigate DDoS attacks, imposing HTTP to HTTPS and so on, requires you to add appropriate libraries within the microservices and manually configure the policies. Instead, you can use the [Rewrite and Responder features](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/crd/rewrite-policy/rewrite-responder-policies-deployment.yaml) provided by the Ingress NetScaler device to deploy these policies. + +Example files: [target-url-rewrite.yaml](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/simplified-deployment-usecases/CRDs/rewrite.md#url-manipulation) + +#### wafs CRD: + +[WAF CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/waf.md) can be used to configure the web application firewall policies with the NetScaler ingress controller on the NetScaler VPX, MPX, SDX, and CPX. The WAF CRD enables communication between the NetScaler ingress controller and NetScaler for enforcing web application firewall policies. + +In a Kubernetes deployment, you can enforce a web application firewall policy to protect the server using the WAF CRD. For more information about web application firewall, see [Web application security](https://docs.netscaler.com/en-us/citrix-adc/current-release/application-firewall/introduction-to-citrix-web-app-firewall.html). + +Example files: [wafhtmlxsssql.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/wafhtmlxsssql.yaml) + +#### CORS CRD: + +[CORS CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/cors.md) Cross-origin resource sharing (CORS) is a mechanism allows a web application running under one domain to securely access resources in another domain. You can configure CORS policies on NetScaler using NetScaler ingress controller to allow one domain (the origin domain) to call APIs in another domain. For more information, see the [cross-origin resource sharing CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/cors.md) documentation. + +Example files: [cors-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/corspolicy-example.yaml) + +#### APPQOE CRD: + +[APPQOE CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/appqoe.md) When a NetScaler appliance receives an HTTP request and forwards it to a back-end server, sometimes there may be connection failures with the back-end server. You can configure the request-retry feature on NetScaler to forward the request to the next available server, instead of sending the reset to the client. Hence, the client saves round trip time when NetScaler initiates the same request to the next available service. +For more information, see the AppQoE support documentation. [Appqoe resource sharing CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/appqoe.md) documentation. + +Example files: [appqoe-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/appqoe_example.yaml) + +#### WILDCARDDNS CRD: + +[WILDCARDDNS CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/wildcarddns.md) Wildcard DNS domains are used to handle requests for nonexistent domains and subdomains. In a zone, use wildcard domains to redirect queries for all nonexistent domains or subdomains to a particular server, instead of creating a separate Resource Record (RR) for each domain. The most common use of a wildcard DNS domain is to create a zone that can be used to forward mail from the internet to some other mail system. +For more information, see the Wild card DNS domains support documentation. [Wildcard DNS Entry CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/wildcarddns.md) documentation. + +Example files: [wildcarddns-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/wildcarddns-example.yaml) + +## NetScaler CPX servicetype LoadBalancer +NetScaler CPX can be installed with service having servicetype LoadBalancer. Following arguments can be used in the `helm install` command for the same: + +``` +helm install netscaler-cpx-with-ingress-controller netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,serviceType.loadBalancer.enabled=True +``` + +## NetScaler CPX servicetype NodePort +NetScaler CPX can be installed with service having servicetype Nodeport. Following arguments can be used in the `helm install` command for the same: + +``` +helm install netscaler-cpx-with-ingress-controller netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,serviceType.nodePort.enabled=True +``` + +Additionally, `serviceType.nodePort.httpPort` and `serviceType.nodePort.httpsPort` arguments can be used to select the nodePort for the CPX service for HTTP and HTTPS ports. + +### Tolerations + +Taints are applied on cluster nodes whereas tolerations are applied on pods. Tolerations enable pods to be scheduled on node with matching taints. For more information see [Taints and Tolerations in Kubernetes](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/). + +Toleration can be applied to pod running NetScaler CPX and ingress controller containers using `tolerations` argument while deploying CPX+NSIC using helm chart. This argument takes list of tolerations that user need to apply on the CPX+NSIC pods. + +For example, following command can be used to apply toleration on the CPX+NSIC pod: + +``` +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,tolerations[0].key=,tolerations[0].value=,tolerations[0].operator=,tolerations[0].effect= +``` + +Here tolerations[0].key, tolerations[0].value and tolerations[0].effect are the key, value and effect that was used while tainting the node. +Effect represents what should happen to the pod if the pod don't have any matching toleration. It can have values `NoSchedule`, `NoExecute` and `PreferNoSchedule`. +Operator represents the operation to be used for key and value comparison between taint and tolerations. It can have values `Exists` and `Equal`. The default value for operator is `Equal`. + +### Resource Quotas +There are various use-cases when resource quotas are configured on the Kubernetes cluster. If quota is enabled in a namespace for compute resources like cpu and memory, users must specify requests or limits for those values; otherwise, the quota system may reject pod creation. The resource quotas for the NSIC and CPX containers can be provided explicitly in the helm chart. + +To set requests and limits for the NSIC container, use the variables `nsic.resources.requests` and `nsic.resources.limits` respectively. +Similarly, to set requests and limits for the CPX container, use the variable `resources.requests` and `resources.limits` respectively. + +Below is an example of the helm command that configures + +A) For NSIC container: + + CPU request for 500milli CPUs + + CPU limit at 1000m + + Memory request for 512M + + Memory limit at 1000M + +B) For CPX container: + + CPU request for 250milli CPUs + + CPU limit at 500m + + Memory request for 256M + + Memory limit at 512M + +``` +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes --set nsic.resources.requests.cpu=500m,nsic.resources.requests.memory=512Mi,nsic.resources.limits.cpu=1000m,nsic.resources.limits.memory=1000Mi --set resources.limits.cpu=500m,resources.limits.memory=512Mi,resources.requests.cpu=250m,resources.requests.memory=256Mi +``` + +### Analytics Configuration +#### Analytics Configuration required for ADM + +If NetScaler CPX needs to send data to the ADM for analytics purpose, then the below steps can be followed to install NetScaler CPX with ingress controller. NSIC configures the NetScaler CPX with the configuration required for analytics. + +1. Create secret using ADM Agent credentials, which will be used by NetScaler CPX to communicate with ADM Agent: + +``` +kubectl create secret generic admlogin --from-literal=username= --from-literal=password= +``` + +|Note: If you have installed container based `adm-agent` using [this](https://github.com/netscaler/netscaler-helm-charts/tree/master/adm-agent) helm chart, above step is not required, you just need to tag the namespace where the CPX is being deployed with `citrix-cpx=enabled`. + +2. Deploy NetScaler CPX with NSIC using helm command: + +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,analyticsConfig.required=true,analyticsConfig.distributedTracing.enable=true,analyticsConfig.endpoint.transactions.service=,ADMSettings.ADMIP=,ADMSettings.loginSecret=,analyticsConfig.transactions.enable=true,analyticsConfig.transactions.port=5557 +``` +|Note: For container based ADM agent, please provide the logstream service FQDN in `analyticsConfig.endpoint.transactions.service`. The `logstream` service will be running on port `5557`. + +#### Analytics Configuration required for NSOE + +If NetScaler CPX needs to send data to the NSOE for observability, then the below steps can be followed to install NetScaler CPX with ingress controller. NSIC configures NetScaler CPX with the configuration required. + +Deploy NetScaler CPX with NSIC using helm command: + +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,analyticsConfig.required=true,analyticsConfig.timeseries.metrics.enable=true,analyticsConfig.timeseries.port=5563,analyticsConfig.timeseries.metrics.mode=prometheus,analyticsConfig.transactions.enable=true,analyticsConfig.transactions.port=5557,analyticsConfig.distributedTracing.enable=true,analyticsConfig.endpoint.metrics.service=,analyticsConfig.endpoint.transactions.service= +``` + +#### Analytics Configuration required for export of metrics to Prometheus + +If NetScaler CPX needs to send data to Prometheus directly without an exporter resource in between, then the below steps can be followed to install NetScaler CPX with ingress controller. NSIC configures NetScaler CPX with the configuration required. + +1. Create secret to enable read-only access for a user, which will be required by NetScaler CPX to export metrics to Prometheus. + +``` +kubectl create secret generic prom-user --from-literal=username= --from-literal=password= +``` + +2. Deploy NetScaler CPX with NSIC using helm command: + +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes,nsic.prometheusCredentialSecret=,analyticsConfig.required=true,analyticsConfig.timeseries.metrics.enable=true,analyticsConfig.timeseries.port=5563,analyticsConfig.timeseries.metrics.mode=prometheus,analyticsConfig.timeseries.metrics.enableNativeScrape=true +``` + +3. To setup Prometheus in order to scrape natively from NetScaler CPX pod, a new scrape job is required to be added under scrape_configs in the Prometheus [configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/). For more details, check kubernetes_sd_config [here](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#kubernetes_sd_config). A sample of the Prometheus job is given below - + +``` + - job_name: 'kubernetes-cpx' + scheme: http + metrics_path: /nitro/v1/config/systemfile + params: + args: ['filename:metrics_prom_ns_analytics_time_series_profile.log,filelocation:/var/nslog'] + format: ['prometheus'] + basic_auth: + username: # Prometheus username set in nsic.prometheusCredentialSecret + password: # Prometheus password set in nsic.prometheusCredentialSecret + scrape_interval: 30s + kubernetes_sd_configs: + - role: pod + relabel_configs: + - source_labels: [__meta_kubernetes_pod_annotation_netscaler_prometheus_scrape] + action: keep + regex: true + - source_labels: [__address__, __meta_kubernetes_pod_annotation_netscaler_prometheus_port] + action: replace + regex: ([^:]+)(?::\d+)?;(\d+) + replacement: $1:$2 + target_label: __address__ + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: kubernetes_namespace + - source_labels: [__meta_kubernetes_pod_name] + action: replace + target_label: kubernetes_pod_name +``` + +> **Note:** +> +> For more details on Prometheus integration, please refer to [this](https://docs.netscaler.com/en-us/citrix-adc/current-release/observability/prometheus-integration) + +### NetScaler CPX License Provisioning +#### Bandwidth based licensing + +By default, CPX runs with 20 Mbps bandwidth called as [CPX Express](https://www.netscaler.com/platform/cpx-container). However, for better performance and production deployments, customer needs licensed CPX instances. [NetScaler ADM](https://docs.netscaler.com/en-us/citrix-application-delivery-management-service/) is used to check out licenses for NetScaler CPX. For more detail on CPX licensing please refer [this](https://docs.netscaler.com/en-us/citrix-adc-cpx/current-release/cpx-licensing.html). + +For provisioning licensing on NetScaler CPX, it is mandatory to provide License Server information to CPX. This can be done by setting **ADMSettings.licenseServerIP** as License Server IP. In addition to this, **ADMSettings.bandWidthLicense** needs to be set true and desired bandwidth capacity in Mbps should be set **ADMSettings.bandWidth**. +For example, to set 2Gbps as bandwidth capacity, below command can be used. + + ``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes --set ADMSettings.licenseServerIP=,ADMSettings.bandWidthLicense=True --set ADMSettings.bandWidth=2000,ADMSettings.licenseEdition="ENTERPRISE" +``` + +#### vCPU based licensing + +For vCPU based licensing on NetScaler CPX, set `ADMSettings.vCPULicense` as True and `ADMSettings.cpxCores` with the number of cores that can be allocated for the CPX. + +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-cpx-with-ingress-controller --set license.accept=yes --set ADMSettings.licenseServerIP=,ADMSettings.vCPULicense=True --set ADMSettings.cpxCores=4,ADMSettings.licenseEdition="ENTERPRISE" +``` + +### Bootup Configuration for NetScaler CPX +To add bootup config on NetScaler CPX, add commands below `cpxCommands` and `cpxShellCommands` in the values.yaml file. The commands will be executed in order. + +For e.g. to add `X-FORWARDED-PROTO` header in all request packets processed by the CPX, add below commands under `cpxCommands` in the `values.yaml` file. + +``` +cpxCommands: | + add rewrite action rw_act_x_forwarded_proto insert_http_header X-Forwarded-Proto "\"https\"" + add rewrite policy rw_pol_x_forwarded_proto CLIENT.SSL.IS_SSL rw_act_x_forwarded_proto + bind rewrite global rw_pol_x_forwarded_proto 10 -type REQ_OVERRIDE +``` + +Commands that needs to be executed in shell of CPX should be kept under `cpxShellCommands` in the `values.yaml` file. + +``` +cpxShellCommands: | + touch /etc/a.txt + echo "this is a" > /etc/a.txt + echo "this is the file" >> /etc/a.txt + ls >> /etc/a.txt +``` + +## Configuration +The following table lists the configurable parameters of the NetScaler CPX with NetScaler ingress controller as side car chart and their default values. + +| Parameters | Mandatory or Optional | Default value | Description | +| ---------- | --------------------- | ------------- | ----------- | +| license.accept | Mandatory | no | Set `yes` to accept the NetScaler ingress controller end user license agreement. | +| imageRegistry | Mandatory | `quay.io` | The NetScaler CPX image registry | +| imageRepository | Mandatory | `netscaler/netscaler-cpx` | The NetScaler CPX image repository | +| imageTag | Mandatory | `14.1-25.111` | The NetScaler CPX image tag | +| pullPolicy | Mandatory | IfNotPresent | The NetScaler CPX image pull policy. | +| daemonSet | Optional | False | Set this to true if NetScaler CPX needs to be deployed as DaemonSet. | +| hostName | Optional | N/A | This entity will be used to set Hostname of the CPX | +| nsic.imageRegistry | Mandatory | `quay.io` | The NetScaler ingress controller image registry | +| nsic.imageRepository | Mandatory | `netscaler/netscaler-k8s-ingress-controller` | The NetScaler ingress controller image repository | +| nsic.imageTag | Mandatory | `2.2.10` | The NetScaler ingress controller image tag | +| nsic.pullPolicy | Mandatory | IfNotPresent | The NetScaler ingress controller image pull policy. | +| nsic.required | Mandatory | true | NSIC to be run as sidecar with NetScaler CPX | +| nsic.enableLivenessProbe| Optional | True | Enable liveness probe settings for NetScaler Ingress Controller | +| nsic.enableReadinessProbe| Optional | True | Enable Readineess probe settings for NetScaler Ingress Controller | +| nsic.livenessProbe | Optional | N/A | Set livenessProbe settings for NSIC | +| nsic.readinessProbe | Optional | N/A | Set readinessProbe settings| +| nsic.resources | Optional | {} | CPU/Memory resource requests/limits for NetScaler Ingress Controller container | +| nsic.rbacRole | Optional | false | To deploy NSIC with RBAC Role set rbacRole=true; by default NSIC gets installed with RBAC ClusterRole(rbacRole=false)) | +| nsic.prometheusCredentialSecret | Optional | N/A | The secret key required to create read only user for native export of metrics using Prometheus. | +| imagePullSecrets | Optional | N/A | Provide list of Kubernetes secrets to be used for pulling the images from a private Docker registry or repository. For more information on how to create this secret please see [Pull an Image from a Private Registry](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). | +| nameOverride | Optional | N/A | String to partially override deployment fullname template with a string (will prepend the release name) | +| fullNameOverride | Optional | N/A | String to fully override deployment fullname template with a string | +| resources | Optional | {} | CPU/Memory resource requests/limits for NetScaler CPX container | +| nitroReadTimeout | Optional | 20 | The nitro Read timeout in seconds, defaults to 20 | +| logLevel | Optional | INFO | The loglevel to control the logs generated by NSIC. The supported loglevels are: CRITICAL, ERROR, WARNING, INFO, DEBUG, TRACE and NONE. For more information, see [Logging](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/configure/log-levels.md).| +| jsonLog | Optional | false | Set this argument to true if log messages are required in JSON format | +| nsConfigDnsRec | Optional | false | To enable/disable DNS address Record addition in NetScaler through Ingress | +| nsSvcLbDnsRec | Optional | false | To enable/disable DNS address Record addition in NetScaler through Type Load Balancer Service | +| nsDnsNameserver | Optional | N/A | To add DNS Nameservers in NetScaler | +| optimizeEndpointBinding | Optional | false | To enable/disable binding of backend endpoints to servicegroup in a single API-call. Recommended when endpoints(pods) per application are large in number. Applicable only for NetScaler Version >=13.0-45.7 | +| defaultSSLCertSecret | Optional | N/A | Provide Kubernetes secret name that needs to be used as a default non-SNI certificate in NetScaler. | +| defaultSSLSNICertSecret | Optional | N/A | Provide Kubernetes secret name that needs to be used as a default SNI certificate in NetScaler. | +| nsHTTP2ServerSide | Optional | OFF | Set this argument to `ON` for enabling HTTP2 for NetScaler service group configurations. | +| cpxLicenseAggregator | Optional | N/A | IP/FQDN of the CPX License Aggregator if it is being used to license the CPX. | +| nsCookieVersion | Optional | 0 | Specify the persistence cookie version (0 or 1). | +| profileSslFrontend | Optional | N/A | Specify the frontend SSL profile. For Details see [Configuration using FRONTEND_SSL_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| profileTcpFrontend | Optional | N/A | Specify the frontend TCP profile. For Details see [Configuration using FRONTEND_TCP_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| profileHttpFrontend | Optional | N/A | Specify the frontend HTTP profile. For Details see [Configuration using FRONTEND_HTTP_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| logProxy | Optional | N/A | Provide Elasticsearch or Kafka or Zipkin endpoint for NetScaler observability exporter. | +| nsProtocol | Optional | http | Protocol http or https used for the communication between NetScaler Ingress Controller and CPX | +| nsEnableLabel | Optional | True | Set to true for plotting Servicegraph. Ensure `analyticsConfig` are set. | +| cpxBgpRouter | Optional | false| If set to true, this CPX is deployed as daemonset in BGP controller mode wherein BGP advertisements are done for attracting external traffic to Kubernetes clusters | +| replicaCount | Optional | 1 | Number of CPX-NSIC pods to be deployed. With `cpxBgpRouter : true`, replicaCount is 1 since CPX will be deployed as DaemonSet | +| nsIP | Optional | 192.168.1.2 | NSIP used by CPX for internal communication when run in Host mode, i.e when cpxBgpRouter is set to true. A /24 internal network is created in this IP range which is used for internal communications withing the network namespace. | +| nsGateway | Optional | 192.168.1.1 | Gateway used by CPX for internal communication when run in Host mode, i.e when cpxBgpRouter is set to true. If not specified, first IP in the nsIP network is used as gateway. It must be in same network as nsIP | +| bgpPort | Optional | 179 | BGP port used by CPX for BGP advertisement if cpxBgpRouter is set to true| +| ingressIP | Optional | N/A | External IP address to be used by ingress resources if not overriden by ingress.com/frontend-ip annotation in Ingress resources. This is also advertised to external routers when pxBgpRouter is set to true| +| entityPrefix | Optional | k8s | The prefix for the resources on the NetScaler CPX. | +| ingressClass | Optional | N/A | If multiple ingress load balancers are used to load balance different ingress resources. You can use this parameter to specify NetScaler ingress controller to configure NetScaler associated with specific ingress class. For more information on Ingress class, see [Ingress class support](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/ingress-classes/). For Kubernetes version >= 1.19, this will create an IngressClass object with the name specified here | +| setAsDefaultIngressClass | Optional | False | Set the IngressClass object as default. New Ingresses without an "ingressClassName" field specified will be assigned the class specified in ingressClass. Applicable only for kubernetes versions >= 1.19 | +| updateIngressStatus | Optional | False | Set this argument if you want to update ingress status of the ingress resources exposed via CPX. | +| disableAPIServerCertVerify | Optional | False | Set this parameter to True for disabling API Server certificate verification. | +| openshift | Optional | false | Set this argument if OpenShift environment is being used. | +| disableOpenshiftRoutes | false | By default Openshift routes are processed in openshift environment, this variable can be used to disable Ingress controller processing the openshift routes. | +| routeLabels | Optional | proxy in () | You can use this parameter to provide the route labels selectors to be used by NetScaler Ingress Controller for routeSharding in OpenShift cluster. | +| namespaceLabels | Optional | N/A | You can use this parameter to provide the namespace labels selectors to be used by NetScaler Ingress Controller for routeSharding in OpenShift cluster. | +| sslCertManagedByAWS | Optional | False | Set this argument if SSL certs used is managed by AWS while deploying NetScaler CPX in AWS. | +| nodeSelector.key | Optional | N/A | Node label key to be used for nodeSelector option for CPX-NSIC deployment. | +| nodeSelector.value | Optional | N/A | Node label value to be used for nodeSelector option in CPX-NSIC deployment. | +| podAnnotations | Optional | N/A | Map of annotations to add to the pods. | +| affinity | Optional | N/A | Affinity labels for pod assignment. | +| tolerations | Optional | N/A | Specify the tolerations for the CPX-NSIC deployment. | +| serviceType.loadBalancer.enabled | Optional | False | Set this argument if you want servicetype of CPX service to be LoadBalancer. | +| serviceType.nodePort.enabled | Optional | False | Set this argument if you want servicetype of CPX service to be NodePort. | +| serviceType.nodePort.httpPort | Optional | N/A | Specify the HTTP nodeport to be used for NodePort CPX service. | +| serviceType.nodePort.httpsPort | Optional | N/A | Specify the HTTPS nodeport to be used for NodePort CPX service. | +| serviceAnnotations | Optional | N/A | Dictionary of annotations to be used in CPX service. Key in this dictionary is the name of the annotation and Value is the required value of that annotation. For example, [see this](#netscaler-adc-cpx-service-annotations). | +| serviceSpec.externalTrafficPolicy | Optional | Cluster | Use this parameter to provide externalTrafficPolicy for CPX service of type LoadBalancer or NodePort. `serviceType.loadBalancer.enabled` or `serviceType.nodePort.enabled` should be set to `true` according to your use case for using this parameter. | +| serviceSpec.loadBalancerIP | Optional | N/A | Use this parameter to provide LoadBalancer IP to CPX service of type LoadBalancer. `serviceType.loadBalancer.enabled` should be set to `true` for using this parameter. | +| serviceSpec.loadBalancerSourceRanges | Optional | N/A | Provide the list of IP Address or range which should be allowed to access the Network Load Balancer. `serviceType.loadBalancer.enabled` should be set to `true` for using this parameter. For details, see [Network Load Balancer support on AWS](https://kubernetes.io/docs/concepts/services-networking/service/#aws-nlb-support). | +| servicePorts | Optional | N/A | List of port. Each element in this list is a dictionary that contains information about the port. For example, [see this](#netscaler-adc-cpx-service-ports). | +| ADMSettings.licenseServerIP | Optional | N/A | Provide the NetScaler Application Delivery Management (ADM) IP address to license NetScaler CPX. For more information, see [Licensing]( https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/licensing/). | +| ADMSettings.licenseServerPort | Optional | 27000 | NetScaler ADM port if non-default port is used. | +| ADMSettings.ADMIP | Optional | N/A | NetScaler Application Delivery Management (ADM) IP address. | +| ADMSettings.loginSecret | Optional | N/A | The secret key to login to the ADM. For information on how to create the secret keys, see [Prerequisites](#prerequistes). | +| ADMSettings.bandWidthLicense | Optional | False | Set to true if you want to use bandwidth based licensing for NetScaler CPX. | +| ADMSettings.bandWidth | Optional | 1000 | Desired bandwidth capacity to be set for NetScaler CPX in Mbps. | +| ADMSettings.vCPULicense | Optional | N/A | Set to true if you want to use vCPU based licensing for NetScaler CPX. | +| ADMSettings.licenseEdition| Optional | PLATINUM | License edition that can be Standard, Platinum and Enterprise . By default, Platinum is selected.| +| ADMSettings.cpxCores | Optional | 1 | Desired number of vCPU to be set for NetScaler CPX. | +| exporter.required | Optional | false | Use the argument if you want to run the [Exporter for NetScaler Stats](https://github.com/netscaler/netscaler-adc-metrics-exporter) along with NetScaler ingress controller to pull metrics for the NetScaler CPX| +| exporter.imageRegistry | Optional | `quay.io` | The Exporter for NetScaler Stats image registry | +| exporter.imageRepository | Optional | `netscaler/netscaler-adc-metrics-exporter` | The Exporter for NetScaler Stats image repository | +| exporter.imageTag | Optional | `1.4.9` | The Exporter for NetScaler Stats image tag | +| exporter.pullPolicy | Optional | IfNotPresent | The Exporter for NetScaler Stats image pull policy. | +| exporter.resources | Optional | {} | CPU/Memory resource requests/limits for Metrics exporter container | +| exporter.ports.containerPort | Optional | 8888 | The Exporter for NetScaler Stats container port. | +| exporter.serviceMonitorExtraLabels | Optional | | Extra labels for service monitor whem NetScaler-adc-metrics-exporter is enabled. | + analyticsConfig.required | Mandatory | false | Set this to true if you want to configure NetScaler to send metrics and transaction records to analytics service. | +| analyticsConfig.distributedTracing.enable | Optional | false | Set this value to true to enable OpenTracing in NetScaler. | +| analyticsConfig.distributedTracing.samplingrate | Optional | 100 | Specifies the OpenTracing sampling rate in percentage. | +| analyticsConfig.endpoint.metrics.service | Optional | N/A | Set this value as the IP address or DNS address of the analytics server. Format: servicename.namespace, servicename.namespace.svc.cluster.local, namespace/servicename *** This value replaces the analyticsConfig.endpoint.server value used earlier. *** | +| analyticsConfig.endpoint.transactions.service | Optional | N/A | Set this value as the IP address or service name with namespace of the analytics service deployed in k8s environment. Format: namespace/servicename *** This value replaces the analyticsConfig.endpoint.service value used earlier. *** | +| analyticsConfig.timeseries.port | Optional | 5563 | Specify the port used to expose analytics service for timeseries endpoint. | +| analyticsConfig.timeseries.metrics.enable | Optional | Set this value to true to enable sending metrics from NetScaler. | +| analyticsConfig.timeseries.metrics.mode | Optional | avro | Specifies the mode of metric endpoint. | +| analyticsConfig.timeseries.metrics.exportFrequency | Optional | 30 | Specifies the time interval for exporting time-series data. Possible values range from 30 to 300 seconds. | +| analyticsConfig.timeseries.metrics.schemaFile | Optional | schema.json | Specifies the name of a schema file with the required Netscaler counters to be added and configured for metricscollector to export. A reference schema file reference_schema.json with all the supported counters is also available under the path /var/metrics_conf/. This schema file can be used as a reference to build a custom list of counters. | +| analyticsConfig.timeseries.metrics.enableNativeScrape | Optional | false | Set this value to true for native export of metrics. | +| analyticsConfig.timeseries.auditlogs.enable | Optional | false | Set this value to true to export audit log data from NetScaler. | +| analyticsConfig.timeseries.events.enable | Optional | false | Set this value to true to export events from the NetScaler. | +| analyticsConfig.transactions.enable | Optional | false | Set this value to true to export transactions from NetScaler. | +| analyticsConfig.transactions.port | Optional | 5557 | Specify the port used to expose analytics service for transaction endpoint. | +| bgpSettings.required | Optional | false | Set this argument if you want to enable BGP configurations for exposing service of Type Loadbalancer through BGP fabric| +| bgpSettings.bgpConfig | Optional| N/A| This represents BGP configurations in YAML format. For the description about individual fields, please refer the [documentation](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/bgp-enhancement.md) | +| nsLbHashAlgo.required | Optional | false | Set this value to set the LB consistent hashing Algorithm | +| nsLbHashAlgo.hashFingers | Optional |256 | Specifies the number of fingers to be used for hashing algorithm. Possible values are from 1 to 1024, Default value is 256 | +| nsLbHashAlgo.hashAlgorithm | Optional | 'default' | Specifies the supported algorithm. Supported algorithms are "default", "jarh", "prac", Default value is 'default' | +| cpxCommands| Optional | N/A | This argument accepts user-provided NetScaler bootup config that is applied as soon as the CPX is instantiated. Please note that this is not a dynamic config, and any subsequent changes to the configmap don't reflect in the CPX config unless the pod is restarted. For more info, please refer the [documentation](https://docs.netscaler.com/en-us/citrix-adc-cpx/current-release/configure-cpx-kubernetes-using-configmaps.html). | +| cpxShellCommands| Optional | N/A | This argument accepts user-provided bootup config that is applied as soon as the CPX is instantiated. Please note that this is not a dynamic config, and any subsequent changes to the configmap don't reflect in the CPX config unless the pod is restarted. For more info, please refer the [documentation](https://docs.netscaler.com/en-us/citrix-adc-cpx/current-release/configure-cpx-kubernetes-using-configmaps.html). | +| enableStartupProbe | Optional | True | Enable startupProbe settings for CPX | +| enableLivenessProbe | Optional | True | Enable livenessProbe settings for CPX | +| startupProbe | Optional | N/A | Set startupProbe settings for CPX | +| livenessProbe | Optional | N/A | Set livenessProbe settings for CPX | + +> **Note:** +> +> If NetScaler ADM related information is not provided during installation, NetScaler CPX will come up with the default license. + +Alternatively, you can define a YAML file with the values for the parameters and pass the values while installing the chart. + +For example: + ``` + helm install netscaler-cpx-with-ingress-controller netscaler/netscaler-cpx-with-ingress-controller -f values.yaml + ``` + +> **Tip:** +> +> The [values.yaml](https://github.com/netscaler/netscaler-helm-charts/blob/master/netscaler-cpx-with-ingress-controller/values.yaml) contains the default values of the parameters. + +## Uninstalling the Chart +To uninstall/delete the ```my-release``` deployment: + ``` + helm delete my-release + ``` + +## Related documentation + +- [NetScaler CPX Documentation](https://docs.netscaler.com/en-us/citrix-adc-cpx/current-release/cpx-architecture-and-traffic-flow) +- [NetScaler ingress controller Documentation](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/) +- [NetScaler ingress controller GitHub](https://github.com/netscaler/netscaler-k8s-ingress-controller) +- [BGP advertisement for External IPs with CPX](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/bgp-enhancement.md) diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/app-readme.md b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/app-readme.md new file mode 100644 index 000000000..6039d0e0c --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/app-readme.md @@ -0,0 +1,5 @@ +# NetScaler CPX with NetScaler Ingress Controller Running as Sidecar + +In a [Kubernetes](https://kubernetes.io/) or [OpenShift](https://www.openshift.com) cluster, you can deploy [NetScaler CPX](https://docs.netscaler.com/en-us/cpx.html) with NetScaler ingress controller as a [sidecar](https://kubernetes.io/docs/concepts/workloads/pods/sidecar-containers/). The NetScaler CPX instance is used for load balancing the North-South traffic to the microservices in your cluster. And, the sidecar NetScaler ingress controller configures the NetScaler CPX. + +This chart bootstraps deployment of NetScaler CPX with NetScaler Ingress Controller as sidecar. diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/crds/crds.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/crds/crds.yaml new file mode 100644 index 000000000..9db6a9972 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/crds/crds.yaml @@ -0,0 +1,2706 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: rewritepolicies.citrix.com +spec: + group: citrix.com + names: + kind: rewritepolicy + plural: rewritepolicies + singular: rewritepolicy + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + rewrite-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to rewrite policy.' + type: array + items: + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be + evaluated if the current policy evaluates to TRUE. + Specify one of the following values: + * NEXT - Evaluate the policy with the next higher priority number. + * END - End policy evaluation. + Default value of goto-priority-expression: END' + type: string + maxLength: 1499 + logpackets: + type: object + description: 'Adds an audit message action. + The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + rewrite-policy: + type: object + properties: + rewrite-criteria: + description: 'Expression against which traffic is evaluated.' + type: string + maxLength: 1299 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). + An UNDEF event indicates an internal error condition.' + type: string + maxLength: 77 + enum: ['NOREWRITE', 'RESET', 'DROP'] + operation: + description: 'Type of user-defined rewrite action.' + type: string + enum: ["noop", "delete", "insert_http_header", "delete_http_header", + "corrupt_http_header", "insert_before", "insert_after", "replace", + "replace_http_res", "delete_all", "replace_all", "insert_before_all", + "insert_after_all", "clientless_vpn_encode", "clientless_vpn_encode_all", + "clientless_vpn_decode", "clientless_vpn_decode_all", "insert_sip_header", + "delete_sip_header", "corrupt_sip_header", "replace_sip_res", "replace_diameter_header_field", + "replace_dns_header_field", "replace_dns_answer_section"] + target: + description: 'Default syntax expression that specifies which part of the request or response to rewrite.' + type: string + maxLength: 1229 + modify-expression: + description: 'Default syntax expression that specifies the content to insert into the request + or response at the specified location, or that replaces the specified string.' + type: string + maxLength: 7991 + multiple-occurence-modify: + description: 'Search facility that is used to match multiple strings in the request or response.' + type: string + maxLength: 171 + additional-multiple-occurence-modify: + description: 'Specify additional criteria to refine the results of the search. + Always starts with the "extend(m,n)" operation, where "m" specifies number of bytes to the left of selected data + and "n" specifies number of bytes to the right of selected data. + You can use refineSearch only on body expressions, and only when rewrite-criteria is any one of this: + INSERT_BEFORE_ALL, INSERT_AFTER_ALL, REPLACE_ALL, and DELETE_ALL.' + type: string + maxLength: 1299 + direction: + description: 'Bind point to which to bind the policy.' + type: string + enum: ["REQUEST","RESPONSE"] + comment: + description: 'Any comments to preserve information about this rewrite policy.' + type: string + maxLength: 255 + required: [rewrite-criteria, operation, target, direction] + required: [rewrite-policy] + + responder-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to responder policy.' + type: array + items: + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be + evaluated if the current policy evaluates to TRUE. + Specify one of the following values: + * NEXT - Evaluate the policy with the next higher priority number. + * END - End policy evaluation. + Default value of goto-priority-expression: END' + type: string + maxLength: 1499 + logpackets: + type: object + description: 'Adds an audit message action. + The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", + "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + responder-policy: + type: object + properties: + redirect: + type: object + description: 'Use this option when you want to Redirect the request when request matches to policy.' + properties: + url: + description: 'URL on which you want to redirect the request.' + type: string + maxLength: 7991 + redirect-status-code: + description: 'HTTP response status code, for example 200, 302, 404, etc.' + type: integer + minimum: 100 + maximum: 599 + redirect-reason: + description: 'Expression specifying the reason for redirecting the request.' + type: string + maxLength: 7991 + required: [url] + respondwith: + type: object + description: 'Use this parameter when you want to respond to the request when request matches to policy.' + properties: + http-payload-string: + description: 'Expression that you want to sent as response to the request.' + type: string + maxLength: 7991 + required: [http-payload-string] + noop: + type: string + description: 'Use this option when you want to send the request to the protected server instead of + responding to it when request matches to policy.' + properties: + target: + description: 'Default syntax expression that specifies to perform noop operation on' + type: string + maxLength: 1229 + reset: + type: string + description: 'Use this option when you want to Reset the client connection by closing it when request matches to policy.' + properties: + drop: + type: string + description: 'Use this option when you want to drop the request without sending a response to the user when request matches to policy.' + properties: + respond-criteria: + description: 'Default syntax expression that the policy uses to determine whether to respond to the specified request.' + type: string + maxLength: 1299 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). + An UNDEF event indicates an internal error condition.' + type: string + maxLength: 77 + enum: ['NOOP', 'RESET', 'DROP'] + comment: + description: 'Any comments to preserve information about this responder policy.' + type: string + maxLength: 255 + required: [respond-criteria] + oneOf: [required: [redirect], required: [respondwith], required: [noop], required: [reset], required: [drop]] + required: [responder-policy] + + dataset: + type: array + items: + type: object + properties: + name: + description: 'Name of the dataset.' + type: string + maxLength: 32 + type: + description: 'Type of value to bind to the dataset.' + type: string + enum: ["ipv4", "number", "ipv6", "ulong", "double", "mac"] + comment: + description: 'Any comments to preserve information about this dataset.' + type: string + maxLength: 255 + values: + description: 'Value of the specified type that is associated with this dataset.' + type: array + items: + type: string + required: [name, type, values] + + patset: + type: array + items: + type: object + properties: + name: + description: 'Name of the Patset.' + type: string + maxLength: 32 + comment: + description: 'Any comments to preserve information about this patset.' + type: string + maxLength: 255 + values: + description: 'String of characters that constitutes a pattern and is associated with this patset.' + type: array + items: + type: string + required: [name, values] + + stringmap: + type: array + items: + type: object + properties: + name: + description: 'Name of the Stringmap.' + type: string + maxLength: 32 + comment: + description: 'Any comments to preserve information about this stringmap.' + type: string + maxLength: 255 + values: + description: 'List of (key,value) pairs to be bound to this string map.' + type: array + items: + type: object + properties: + key: + description: 'Character string constituting the key to be bound to this string map.' + type: string + maxLength: 2047 + value: + description: 'Character string constituting the value associated with the key.' + type: string + maxLength: 2047 + required: [name, values] + + httpcallout_policy: + type: array + items: + type: object + properties: + name: + description: 'httpcallout name' + type: string + maxLength: 32 + server_ip: + description: 'IP Address of the server(callout agent) to which the callout is sent.' + type: string + server_port: + description: 'Port of the server(callout agent) to which the callout is sent.' + type: integer + minimum: 1 + maximum: 65535 + http_method: + description: |+ + 'Method used in the HTTP request that this callout sends. + Default http method is GET' + type: string + enum: ['GET', 'POST'] + host_expr: + description: |+ + 'String expression to configure the Host header. Can contain a literal value + (for example, 10.101.10.11) or a derived value (for example, http.req.header("Host")). + The literal value can be an IP address or a fully qualified domain name. Mutually + exclusive with the full HTTP request expression.' + type: string + maxLength: 255 + url_stem_expr: + description: |+ + 'String expression for generating the URL stem. Can contain a literal string + (for example, "/mysite/index.html") or an expression that derives the value + (for example, http.req.url).' + type: string + maxLength: 8191 + headers: + type: array + description: |+ + 'One or more headers to insert into the HTTP request. Each header is represented by + name and expr, where expr is an expression that is evaluated at runtime to provide + the value for the named header. You can configure a maximum of eight headers for + an HTTP callout.' + items: + type: object + properties: + name: + description: 'header name' + type: string + expr: + description: 'header expression' + type: string + parameters: + type: array + description: |+ + 'One or more query parameters to insert into the HTTP request URL (for a GET request) + or into the request body (for a POST request). Each parameter is represented by + name and expr, where expr is an expression that is evaluated at run time to provide + the value for the named parameter (name=value). The parameter values are URL encoded.' + items: + type: object + properties: + name: + description: 'parameter name' + type: string + expr: + description: 'parameter expression' + type: string + body_expr: + description: |+ + 'An advanced string expression for generating the body of the request. + The expression can contain a literal string or an expression that derives + the value (for example, client.ip.src).' + type: string + full_req_expr: + description: |+ + 'Exact HTTP request, in the form of an expression, which the NetScaler sends + to the callout agent. The request expression is constrained by the feature + for which the callout is used. For example, an HTTP.RES expression cannot be + used in a request-time policy bank or in a TCP content switching policy bank.' + type: string + scheme: + description: |+ + 'Type of scheme for the callout server. + Default scheme is HTTP' + type: string + enum: ['HTTP', 'HTTPS'] + cache_for_secs: + description: |+ + 'Duration, in seconds, for which the callout response is cached. + The cached responses are stored in an integrated caching content + group named "calloutContentGroup". If no duration is configured, + the callout responses will not be cached unless normal caching + configuration is used to cache them. This parameter takes precedence over any + normal caching configuration that would otherwise apply to these responses.' + type: integer + minimum: 1 + maximum: 31536000 + return_type: + description: |+ + 'Type of data that the target callout agent returns in response to the callout + Available settings function as follows: + * TEXT - Treat the returned value as a text string. + * NUM - Treat the returned value as a number. + * BOOL - Treat the returned value as a Boolean value.' + type: string + enum: ['TEXT', 'NUM', 'BOOL'] + result_expr: + description: |+ + 'Expression that extracts the callout results from the response sent by the HTTP callout + agent. Must be a response based expression, that is, it must begin with HTTP.RES. The + operations in this expression must match the return type. For example, if you configure + a return type of TEXT, the result expression must be a text based expression. If the + return type is NUM, the result expression (resultExpr) must return a numeric value, + as in the following example: http.res.body(10000).length.' + type: string + maxLength: 8191 + comment: + description: 'Any comments to preserve information about this HTTP callout.' + type: string + maxLength: 255 + allOf: + - properties: + required: [name, server_ip, server_port] + - properties: + oneOf: + - properties: + required: [full_req_expr] + - properties: + anyOf: + - properties: + required: [http_method] + - properties: + required: [host_expr] + - properties: + required: [url_stem_expr] + - properties: + required: [headers] + - properties: + required: [parameters] + - properties: + required: [body_expr] + anyOf: [required: [rewrite-policies], required: [responder-policies]] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ratelimits.citrix.com +spec: + group: citrix.com + names: + kind: ratelimit + plural: ratelimits + singular: ratelimit + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the ratelimit policies are applied.' + type: array + items: + type: string + maxLength: 127 + selector_keys: + type: object + description: 'Traffic match criteria to which apply above rate-limit/throttling. All keys are applied as AND condition. If no keys are specified, rate-limit applies at service level' + properties: + basic: + type: object + description: "Basic traffic stream selection criteria to which to apply the ratelimit" + properties: + path: + type: array + description: "api resource path prefix match. e.g. /api/v1/products" + items: + type: string + method: + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header_name: + description: "HTTP header that identifies the unique API client for e.g. X-apikey" + type: string + per_client_ip: + description: "Setting this applies the throttling limit to each unique Client IP address accessing the API resource" + type: boolean + req_threshold: + description: 'Max requests per timeslice units to be allowed' + type: integer + timeslice: + description: 'Timeslice in miliseconds in multiple of 10. Defaults to 1000 miliseconds' + type: integer + limittype: + description: "Burst mode or smooth. Defaults to smooth limittype if not specified" + type: string + enum: ['BURSTY','SMOOTH'] + throttle_action: + type: string + enum: ['DROP', 'RESET','REDIRECT', 'RESPOND'] + description: "Drop will drop the requests exceeding limits, RESET will reset the client connection, Redirect will redirect to specified URL, respond will respond with 429 'Exceeded allowed rate of requests'" + redirect_url: + type: string + description: "Redirect-URL" + logpackets: + type: object + description: 'Adds an audit message action. The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + required: [req_threshold] +--- +#Sample CRD instance + +#apiVersion: citrix.com/v1 +#description: VIP for apache service +#kind: vip +#metadata: +# name: service-apache +# namespace: default +#spec: +# description: VIP for the apache Service +# ipaddress: 10.99.98.90 +# kind: service +# name: apache + +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vips.citrix.com +spec: + group: citrix.com + names: + kind: vip + plural: vips + singular: vip + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + additionalPrinterColumns: + - jsonPath: .spec.ipaddress + name: VIP + type: string + - name: Age + type: date + jsonPath: .metadata.creationTimestamp + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + ipaddress: + type: string + name: + type: string + kind: + type: string + enum: ["service", "ingress", "listener"] + description: + type: string + range-name: + type: string + multicluster: + description: "The setting of this indicates that the VIP/csvserver IP address is shared by multiple netscaler ingress controllers on the VPX/MPX. For CPX, this field is not applicable" + type: boolean +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: authpolicies.citrix.com +spec: + group: citrix.com + names: + kind: authpolicy + plural: authpolicies + singular: authpolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: 'Current Status of the CRD' + jsonPath: .status.state + - name: Message + type: string + description: 'Status Message' + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: |+ + 'Name of the services for which the policies applied' + type: array + items: + type: string + maxLength: 63 + authentication_mechanism: + type: object + description: |+ + 'Authentication mechanism. Options: using forms or using request header. + Default is Authentication using request header, when no option is specified' + properties: + using_request_header: + description: |+ + 'Enable user authentication using request header. Use when the credentials + or api keys are passed in a header. For example, when using Basic, Digest, + Bearer authentication or api keys. + When authentication using forms is provided, this is set to OFF' + + type: string + using_forms: + type: object + description: 'Enables authentication using forms. Use with user/web authentication.' + properties: + authentication_host: + description: |+ + 'Fully qualified domain name (FQDN) for authentication. + This FQDN should be unique and should resolve to frontend IP of + NetScaler with Ingress/service type LoadBalancer (or) vip of Listener CRD' + type: string + maxLength: 255 + authentication_host_cert: + description: |+ + 'Name of the SSL certificate to be used with authentication_host. + This certificate is mandatory while using_forms' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - properties: + required: [tls_secret] + - properties: + required: [preconfigured] + ingress_name: + description: |+ + 'Ingress name for which the authentication using forms + is applicable.' + type: string + maxLength: 63 + lb_service_name: + description: |+ + 'Service of type LoadBalancer for which the authentication using forms + is applicable.' + type: string + maxLength: 63 + listener_name: + description: |+ + 'Listener CRD name for which the authentication using forms is applicable.' + type: string + maxLength: 63 + vip: + description: |+ + 'Frontend IP of ingress for which the authentication + using forms is applicable. This refers to frontend-ip provided + with Ingress. It is suggested to use vip, if more than one Ingress + resource use the same frontend-ip' + type: string + required: [authentication_host, authentication_host_cert] + oneOf: + - properties: + required: [ingress_name] + - properties: + required: [lb_service_name] + - properties: + required: [listener_name] + - properties: + required: [vip] + oneOf: + - properties: + using_request_header: + enum: ['ON'] + required: [using_request_header] + - properties: + required: [using_forms] + + authentication_providers: + description: |+ + 'Authentication Configuration for required authentication providers/schemes. + One or more of these can be created' + type: array + items: + description: 'Create config for a single authentication provider of a particular type' + type: object + properties: + name: + description: 'Name for this provider, has to be unique, referenced by authentication policies' + type: string + maxLength: 127 + + oauth: + description: 'Authentication provided by external oAuth provider' + type: object + properties: + issuer: + description: 'Identity of the server whose tokens are to be accepted' + type: string + maxLength: 127 + audience: + description: 'Audience for which token sent by Authorization server is applicable' + type: array + items: + type: string + maxLength: 127 + jwks_uri: + description: |+ + 'URL of the endpoint that contains JWKs (Json Web Key) for + JWT (Json Web Token) verification' + type: string + maxLength: 127 + introspect_url: + description: ' URL of the introspection server' + type: string + maxLength: 127 + client_credentials: + description: |+ + 'secrets object that contains Client Id and secret as known + to Introspection server' + type: string + maxLength: 253 + token_in_hdr: + description: |+ + 'custom header name where token is present, + default is Authorization header' + type: array + items: + type: string + maxLength: 127 + maxItems: 2 + token_in_param: + description: 'query parameter name where token is present' + type: array + items: + type: string + maxLength: 127 + maxItems: 2 + signature_algorithms: + description: 'list of allowed signature algorithms, by default HS256, RS256, RS512 are allowed' + type: array + items: + type: string + enum: ['HS256', 'RS256', 'RS512'] + claims_to_save: + description: 'list of claims to be saved, used to create authorization policies' + type: array + items: + type: string + maxLength: 127 + metadata_url: + description: 'URL used to get OAUTH/OIDC provider metadata' + type: string + maxLength: 255 + user_field: + description: |+ + 'Attribute in the token from which username should be extracted. + by default, NetScaler looks at email attribute for user id' + type: string + maxLength: 127 + default_group: + description: |+ + 'group assigned to the request if authentication succeeds, + this is in addition to any extracted groups from token' + type: string + maxLength: 63 + grant_type: + description: 'used to specify the type of flow to the token end point, defaults to CODE' + type: array + items: + type: string + enum: ['CODE','PASSWORD'] + pkce: + description: 'specify whether to enable Proof Key Code Exchange, defaults to ENABLED' + type: string + enum: ['ENABLED', 'DISABLED'] + token_ep_auth_method: + description: |+ + 'authentication method to be used with token end point, + defaults to client_secret_post' + type: string + enum: ['client_secret_post', 'client_secret_jwt'] + + anyOf: + - properties: + required : [jwks_uri] + - properties: + required : [introspect_url, client_credentials] + - properties: + required : [metadata_url] + + ldap: + description: 'LDAP authentication provider' + type: object + properties: + server_ip: + description: 'IP address assigned to the LDAP server' + type: string + server_name: + description: 'LDAP server name as a FQDN' + type: string + maxLength: 127 + server_port: + description: 'Port on which the LDAP server accepts connections. Default is 389' + type: integer + minimum: 1 + maximum: 65535 + base: + description: |+ + 'Base (node) from which to start LDAP searches. If the LDAP server is + running locally, the default value of base is dc=netscaler, dc=com' + type: string + maxLength: 127 + server_login_credentials: + description: |+ + 'Kubernetes secret object providing credentials to login to LDAP server, + The secret data should have username and password' + type: string + login_name: + description: |+ + 'LDAP login name attribute. The NetScaler uses the LDAP login name + to query external LDAP servers or Active Directories' + type: string + maxLength: 127 + security_type: + description: |+ + 'Type of security used for communications between the NetScaler + and the LDAP server. Default is TLS' + type: string + enum: ['PLAINTEXT', 'TLS', 'SSL'] + validate_server_cert: + description: 'Validate LDAP Server certs. Default is NO' + type: string + enum: ['YES', 'NO'] + hostname: + description: |+ + 'Hostname for the LDAP server. If validate_server_cert is ON, + this must be the host name on the certificate from the LDAP + A hostname mismatch will cause a connection failure' + type: string + maxLength: 127 + sub_attribute_name: + description: 'LDAP group sub-attribute name. Used for group extraction from the LDAP server.' + type: string + maxLength: 31 + group_attribute_name: + description: 'LDAP group attribute name. Used for group extraction on the LDAP server.' + type: string + maxLength: 31 + search_filter: + description: |+ + 'String to be combined with the default LDAP user search string to form the + search value. For example, if the search filter "vpnallowed=true" is combined + with the LDAP login name "samaccount" and the user-supplied username is "bob", + the result is the LDAP search string ""(&(vpnallowed=true)(samaccount=bob)"" + (Be sure to enclose the search string in two sets of double quotation marks)' + type: string + maxLength: 255 + auth_timeout: + description: |+ + 'Number of seconds the NetScaler waits for a response from the server + Default is 3' + type: integer + minimum: 1 + maximum: 4294967295 + password_change: + description: 'Allow password change requests. Default is DISABLED' + type: string + enum: ['ENABLED', 'DISABLED'] + attributes_to_save: + description: |+ + 'List of attribute names separated by comma which needs to be fetched + from LDAP server and stored as key-value pair for the session on NetScaler' + type: string + maxLength: 2047 + oneOf: + - properties: + required: [server_ip] + - properties: + required: [server_name] + + saml: + description: |+ + 'SAML authentication provider. + Currently SAML is supported only with authentication mechanism using forms' + type: object + properties: + metadata_url: + description: 'URL is used for obtaining saml metadata.' + type: string + maxLength: 255 + metadata_refresh_interval: + description: |+ + 'Interval in minutes for fetching metadata from specified metadata URL. + Default is 36000' + type: integer + minimum: 1 + maximum: 4294967295 + signing_cert: + description: 'SSL certificate to sign requests from SP to IDP' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - properties: + required: [tls_secret] + - properties: + required: [preconfigured] + audience: + description: 'Audience for which assertion sent by IdP is applicable' + type: string + maxLength: 127 + issuer_name: + description: 'The name to be used in requests sent from SP to IDP to identify NetScaler' + type: string + maxLength: 63 + binding: + description: 'Specifies the transport mechanism of saml message. Default is POST' + type: string + enum: ['REDIRECT', 'POST', 'ARTIFACT'] + artifact_resolution_service_url: + description: 'URL of the Artifact Resolution Service on IdP' + type: string + maxLength: 255 + logout_binding: + description: 'Specifies the transport mechanism of saml logout. Default is POST' + type: string + enum: ['REDIRECT', 'POST'] + reject_unsigned_assertion: + description: |+ + 'Reject unsigned SAML assertions. ON, rejects assertion without signature. + STRICT ensure that both Response and Assertion are signed. Default is ON' + type: string + enum: ['ON', 'OFF', 'STRICT'] + user_field: + description: 'SAML user ID, as given in the SAML assertion' + type: string + maxLength: 63 + default_authentication_group: + description: |+ + 'This is the default group that is chosen when the authentication + succeeds in addition to extracted groups' + type: string + maxLength: 63 + skew_time: + description: |+ + 'Allowed clock skew in number of minutes on an incoming assertion. + Default is 5' + type: integer + minimum: 1 + attributes_to_save: + description: |+ + 'List of attribute names separated by comma which needs to be extracted + and stored as key-value pair for the session on NetScaler' + type: string + maxLength: 2047 + required: + - metadata_url + + basic_local_db: + type: object + description: |+ + 'Basic HTTP authentication supported by NetScaler, user data in local DB of NetScaler. + Users needs to be added on NetScaler' + properties: + use_local_auth: + description: 'Use NetScaler authentication' + type: string + enum: ['YES'] + + required: + - name + + authentication_policies: + description: 'Authentication policies' + type: array + items: + type: object + description: 'Authentication policy' + properties: + resource: + type: object + description: 'endpoint/resource selection criteria' + properties: + path: + description: 'api resource path e.g. /products. ' + type: array + items: + type: string + maxLength: 511 + method: + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + required: + - path + expression: + description: 'NetScaler syntax expression for authentication' + type: string + maxLength: 1229 + provider: + description: 'name of the authentication provider for the policy, empty if no authentication required' + type: array + items: + type: string + maxLength: 127 + maxItems: 1 + oneOf: + - required: [resource, provider] + - required: [expression, provider] + + authorization_policies: + description: 'Authorization policies' + type: array + items: + type: object + description: 'Authorization policy' + properties: + resource: + type: object + description: 'endpoint/resource selection criteria' + properties: + path: + description: 'api resource path e.g. /products. ' + type: array + items: + type: string + maxLength: 511 + method: + description: ' http method' + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + claims: + description: 'authorization scopes required for selected resource saved as claims or attributes' + type: array + items: + type: object + properties: + name: + description: 'name of the claim/attribute to check' + type: string + maxLength: 127 + values: + description: 'list of claim values required for the request' + type: array + items: + type: string + maxLength: 127 + minItems: 1 + required: + - name + - values + required: + - claims + expression: + description: 'NetScaler syntax expression for authorization' + type: string + maxLength: 1229 + oneOf: + - required: [resource] + - required: [expression] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: listeners.citrix.com +spec: + group: citrix.com + names: + kind: Listener + plural: listeners + singular: listener + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + required: [spec] + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + required: [protocol] + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + multicluster: + description: "The setting of this indicates that the VIP/csvserver IP address is shared by multiple netscaler ingress controllers on the VPX/MPX. For CPX, this field is not applicable" + type: boolean + protocol: + type: string + enum: ["udp", "tcp", "https", "http"] + description: "Protocol for this listener" + vip: + type: string + description: "VIP address, Optional for CPX, required for Tier-1 deployments" + secondaryVips: + type: array + description: "An array of Secondary VIPs. All the VIPs will be part of an ipset" + minItems: 1 + items: + type: string + redirectPort: + type: integer + minimum: 1 + maximum: 65535 + description: "Port from which http traffic should be redirected to https" + port: + type: integer + minimum: 1 + maximum: 65535 + certificates: + type: array + description: "certificates attached to the endpoints - Not applicable for HTTP" + minItems: 1 + items: + type: object + properties: + preconfigured: + type: string + description: "Preconfigured Certificate name on NetScaler " + secret: + type: object + description: "Kuberentes secret object" + required: [name] + properties: + name: + type: string + description: "name of the Kubernetes Secret object where Cert is located" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + type: string + description: "Namespace of the kubernetes secret object; Default is same namespace where the Listener object is located" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + default: + type: boolean + description: "Only one of the certificate can be marked as default which will be presented if none of the cert matches with the hostname" + oneOf: + - required: ["preconfigured"] + - required: ["secret"] + policies: + type: object + description: "Policies attached to the Listener" + properties: + httpprofile: + type: object + description: "HTTP profile configurations for the Listener, HTTP level configurations" + properties: + preconfigured: + type: string + description: "Preconfigured or Built-in HTTP profile name" + config: + type: object + description: "HTTP profile configuration for the listener. For individual fields, refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ns/nshttpprofile Name field is auto populated" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + tcpprofile: + type: object + description: "TCP level configurations, uses ns tcpprofile of NetScaler" + properties: + preconfigured: + description: "Preconfigured or Built-in TCP profile name" + type: string + config: + type: object + description: "TCPprofile configurations for the listener. For individual fields refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ns/nstcpprofile ; Name field is auto populated" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + csvserverConfig: + type: object + description: "CS Vserver configuration for the listener" + additionalProperties: + type: string + sslprofile: + type: object + description: "SSL profile configuration" + properties: + preconfigured: + type: string + description: "SSL profile which is preconfigured in NetScaler. Ciphers bound to the profile is not overriden" + config: + description: "NetScaler frontend SSL profile configurations. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ssl/sslprofile.html for all configurations; Name field is auto generated" + type: object + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + sslciphers: + type: array + description: "List of ciphers to be bound to the ssl profile for the listener. Priority is as per the order in the list. A cipher suite, predefined cipher group or User created cipher group can be mentioned" + minItems: 1 + items: + type: string + description: "Cipher suite, cipher group name" + analyticsprofile: + type: object + description: "Analytics profile configuration" + properties: + preconfigured: + type: array + description: "Preconfigured Analytics profile that needs to be bound to the vserver" + minItems: 1 + items: + type: string + description: "Name of the analytics profile preconfigured that will be bound to the Vserver" + config: + type: array + description: "An array of analytics to be enabled" + minItems: 1 + items: + type: object + description: "Anlytics to be enabled" + required: ['type'] + properties: + type: + description: "Analytics profile to be enabled, you can enable one or more of the webinsight, tcpinsight, securityinsight, videoinsight, hdxinsight, gatewayinsight, timeseries, lsninsight, botinsight " + type: string + enum: ["webinsight", "tcpinsight", "securityinsight", "videoinsight", "hdxinsight", "gatewayinsight", "timeseries", "lsninsight", "botinsight"] + parameters: + type: object + description: "Additional parameters for analytics profile. Please refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/analytics/analyticsprofile/" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + routes: + type: array + description: "List of route objects attached to the listener" + minItems: 1 + items: + type: object + properties: + name: + type: string + description: "Name of the HTTPRoute object" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + type: string + description: "Namespace of the HTTPRoute object" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + labelSelector: + description: "Labels key value pair, if the route carries the same labels, it is automatically attached" + type: object + additionalProperties: + type: string + oneOf: + - required: [name, namespace] + - required: [labelSelector] + defaultAction: + type: object + description: "Default action for the listener: One of Backend or Redirect" + properties: + backend: + type: object + oneOf: + - required: [kube] + properties: + kube: + type: object + required: [service, port] + properties: + service: + description: "Name of the backend service" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + port: + description: "Service port" + type: integer + minimum: 1 + maximum: 65535 + namespace: + description: "Service namespace" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + backendConfig: + description: "General backend service options" + type: object + properties: + secure_backend: + description: "Use Secure communications to the backends" + type: boolean + lbConfig: + description: "NetScaler LB vserver configurations for the backend. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/lb/lbvserver.html for all configurations" + type: object + additionalProperties: + type: string + servicegroupConfig: + description: "NetScaler service group configurations for the backend; Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/basic/servicegroup.html for all configurations" + type: object + additionalProperties: + type: string + redirect: + type: object + oneOf: + - required: [targetExpression] + - required: [hostRedirect] + - required: [httpsRedirect] + properties: + httpsRedirect: + description: "Change the scheme from http to https keeping URL intact" + type: boolean + hostRedirect: + description: "Host name specified is used for redirection with URL intact" + type: string + targetExpression: + description: "A target can be specified using NetScaler policy expression" + type: string + responseCode: + description: "Default response code is 302, which can be customised using this attribute" + type: integer + minimum: 100 + maximum: 599 + oneOf: + - required: ["backend"] + - required: ["redirect"] + subresources: + # status enables the status subresource. + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: httproutes.citrix.com +spec: + group: citrix.com + names: + kind: HTTPRoute + plural: httproutes + singular: httproute + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + required: [rules] + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + hostname: + type: array + description: "List of domain names that share the same route, default is '*'" + minItems: 1 + items: + type: string + description: "Domain name" + rules: + type: array + description: "List Content routing rules with an action defined" + minItems: 1 + items: + type: object + required: [name, action] + properties: + name: + type: string + description: "A name to represent the rule, this is used as an identifier in content routing policy name in NetScaler" + minLength: 1 + maxLength: 20 + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + match: + type: array + description: "List of rules with same action" + minItems: 1 + items: + type: object + anyOf: + - required: [path] + - required: [headers] + - required: [cookies] + - required: [queryParams] + - required: [method] + - required: [policyExpression] + properties: + path: + type: object + description: "URL Path based content routing" + properties: + prefix: + type: string + description: "URL path matches the prefix expression" + exact: + type: string + description: "URL Path must match exact path" + regex: + type: string + description: "PCRE based regex expression for path matching" + headers: + type: array + description: "List of header for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Header details for content routing, Check for existence of a header or header name-value match" + properties: + headerName: + type: object + description: "Header name based content routing, Here existence of header is used for routing" + properties: + exact: + type: string + description: "Header Name - treated as exact must exist" + contains: + type: string + description: "Header Name - A header must exist that contain the string the name" + regex: + type: string + description: "header Name - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e header name must not exist" + oneOf: + - required: [exact] + - required: [contains] + - required: [regex] + headerValue: + type: object + description: "Header Name and Value based match" + properties: + name: + type: string + description: "Header name that must match the value" + exact: + type: string + description: "Header value - treated as exact" + contains: + type: string + description: "Header value - treated as contains" + regex: + type: string + description: "header value - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e header if present must not match the value" + oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + queryParams: + type: array + description: "List of Query parameters for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Query parameters Name and Value based match" + properties: + name: + type: string + description: "Query name that must match the value. If no value is specified, matches with any value" + exact: + type: string + description: "Query value - Exact match" + contains: + type: string + description: "Query value - value must have the string(substring)" + regex: + type: string + description: "Query value - Value must match this regex patterm" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e query if present must not match the value" + anyOf: + - required: [name] + - oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + cookies: + type: array + description: "List of Cookie params for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Cookie based routing" + properties: + name: + type: string + description: "cookie name that must match the value. If no value specified, it matches with any value" + exact: + type: string + description: "cookie value - treated as exact" + contains: + type: string + description: "cookie value - treated as substring" + regex: + type: string + description: "cookie value - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e cookie if present must not match the value" + anyOf: + - required: [name] + - oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + method: + type: string + description: "HTTP method for content routing eg: POST, PUT, DELETE etc" + policyExpression: + type: string + description: "NetScaler policy expressions; refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/policy/policyexpression.html" + action: + type: object + description: "Action for the matched rule" + properties: + backend: + type: object + oneOf: + - required: [kube] + properties: + kube: + type: object + required: [service, port] + properties: + service: + description: "Name of the backend service" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + port: + description: "Service port" + type: integer + minimum: 1 + maximum: 65535 + backendConfig: + type: object + description: "General backend service options" + properties: + secureBackend: + description: "Use Secure communications to the backends" + type: boolean + lbConfig: + description: "NetScaler LB vserver configurations for the backend. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/lb/lbvserver.html for all configurations" + type: object + additionalProperties: + type: string + servicegroupConfig: + description: "NetScaler service group configurations for the backend; Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/basic/servicegroup.html for all configurations" + type: object + additionalProperties: + type: string + redirect: + type: object + oneOf: + - required: [targetExpression] + - required: [hostRedirect] + - required: [httpsRedirect] + properties: + httpsRedirect: + description: "Change the scheme from http to https keeping URL intact" + type: boolean + hostRedirect: + description: "Host name specified is used for redirection with URL intact" + type: string + targetExpression: + description: "A target can be specified using NetScaler policy expression" + type: string + responseCode: + description: "Default response code is 302, which can be customised using this attribute" + type: integer + minimum: 100 + maximum: 599 + oneOf: + - required: ["backend"] + - required: ["redirect"] + subresources: + # status enables the status subresource. + status: {} + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + # name must match the spec fields below, and be in the form: . + name: continuousdeployments.citrix.com +spec: + group: citrix.com + names: + kind: continuousdeployment + plural: continuousdeployments + singular: continuousdeployment + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + cronSpec: + type: integer + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: wafs.citrix.com +spec: + group: citrix.com + names: + kind: waf + plural: wafs + singular: waf + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the waf policies are applied.' + type: array + items: + type: string + maxLength: 127 + application_type: + description: 'Type of applications to protect' + type: array + items: + type: string + enum: ['HTML', 'JSON', 'XML'] + signatures: + description: 'Location of external signature file' + type: string + redirect_url: + description: 'When a URL is blocked/down, redirect_url represents the alternate URL where the client requests should be sent.' + type: string + html_error_object: + description: 'Location of customized error page to respond when html or common violation are hit' + type: string + xml_error_object: + description: 'Location of customized error page to respond when xml violations are hit' + type: string + json_error_object: + description: 'Location of customized error page to respond when json violations are hit' + type: string + ip_reputation: + type: object + x-kubernetes-preserve-unknown-fields: true + description: 'Enabling IP reputation feature' + target: + description: 'To control what traffic to be inspected by Web Application Firewall. If you do not provide the target, everything will be inspected by default' + type: object + properties: + path: + type: array + description: "List of http urls to inspect" + items: + type: string + description: "URL path" + method: + type: array + description: "List of http methods to inspect" + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header: + type: array + description: "List of http headers to inspect" + items: + type: string + description: "header name" + security_checks: + description: 'To enable/disable application firewall security checks' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + settings: + description: 'To fine tune application firewall security checks default settings' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + relaxations: + description: 'Section which contains relaxation rules for known traffic and false positives' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + enforcements: + description: 'Section which contains enforcement or restriction rules' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: bots.citrix.com +spec: + group: citrix.com + names: + kind: bot + plural: bots + singular: bot + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the bot policies are applied.' + type: array + items: + type: string + maxLength: 127 + signatures: + description: 'Location of external bot signature file' + type: string + redirect_url: + description: 'url to redirect when bot violation is hit' + type: string + target: + description: 'To control what traffic to be inspected by BOT. If you do not provide the target, everything will be inspected by default' + type: object + properties: + path: + type: array + description: "List of http urls to inspect" + items: + type: string + description: "URL path" + method: + type: array + description: "List of http methods to inspect" + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header: + type: array + description: "List of http headers to inspect" + items: + type: string + description: "header name" + security_checks: + description: 'To enable/disable bot ecurity checks' + type: object + properties: + allow_list: + type: string + enum: ['ON', 'OFF'] + block_list: + type: string + enum: ['ON', 'OFF'] + device_fingerprint: + type: string + enum: ['ON', 'OFF'] + device_fingerprint_action: + type: object + x-kubernetes-preserve-unknown-fields: true + headless_browser: + type: string + enum: ['ON','OFF'] + reputation: + type: string + enum: ['ON', 'OFF'] + ratelimit: + type: string + enum: ['ON', 'OFF'] + tps: + type: string + enum: ['ON', 'OFF'] + trap: + type: object + x-kubernetes-preserve-unknown-fields: true + bindings: + description: 'Section which contains binding rules for bot security checks' + type: object + properties: + allow_list: + type: array + items: + type: object + properties: + subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6_subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + expression: + type: object + x-kubernetes-preserve-unknown-fields: true + + block_list: + type: array + items: + type: object + properties: + subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6_subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + expression: + type: object + x-kubernetes-preserve-unknown-fields: true + ratelimit: + type: array + items: + type: object + properties: + url: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + cookie: + type: object + x-kubernetes-preserve-unknown-fields: true + geolocation: + type: object + x-kubernetes-preserve-unknown-fields: true + reputation: + type: object + x-kubernetes-preserve-unknown-fields: true + captcha: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + logexp: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + kbmexpr: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + tps: + type: object + properties: + geolocation: + type: object + x-kubernetes-preserve-unknown-fields: true + host: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + url: + type: object + x-kubernetes-preserve-unknown-fields: true + trapinsertion: + type: object + x-kubernetes-preserve-unknown-fields: true +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: apigatewaypolicies.citrix.com +spec: + group: citrix.com + names: + kind: apigatewaypolicy + plural: apigatewaypolicies + singular: apigatewaypolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + api_definition: + type: object + properties: + repository: + type: string + branch: + type: string + oas_secret_ref: + type: string + files: + type: array + items: + type: string + maxLength: 127 + api_proxy: + type: object + properties: + ipaddress: + type: string + port: + type: integer + protocol: + type: string + secret: + type: string + policies: + type: array + items: + type: object + properties: + name: + type: string + selector: + type: array + items: + type: object + properties: + tags: + type: array + items: + type: string + api: + type: string + method: + type: array + items: + type: string + maxLength: 127 + upstream: + type: object + properties: + service: + type: string + port: + type: integer + policy_bindings: + type: object + properties: + ratelimit: + type: object + properties: + name: + type: string + waf: + type: object + properties: + name: + type: string + rewritepolicy: + type: object + properties: + name: + type: string + bot: + type: object + properties: + name: + type: string + aaa: + type: array + items: + type: object + properties: + crd_name: + type: string + mappings: + type: array + items: + type: object + properties: + petstore_auth: + type: string + api_key: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: corspolicies.citrix.com +spec: + group: citrix.com + names: + kind: corspolicy + plural: corspolicies + singular: corspolicy + shortNames: + - cp + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: 'Current Status of the CRD' + jsonPath: .status.state + - name: Message + type: string + description: 'Status Message' + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'The list of Kubernetes services to which you want to apply the cors policies.' + type: array + items: + type: string + maxLength: 63 + allow_origin: + description: 'Represents list of allowed origins, it is used to screen the “origin” in the cors pre flight request' + type: array + items: + type: string + maxLength: 2083 + allow_methods: + description: 'Indicates which methods are supported by the response’s URL for the purposes of the CORS protocol. This variable will be used to set Access-Control-Allow-Methods in the pre-flight cors response.' + type: array + items: + type: string + maxLength: 127 + allow_headers: + description: 'Indicates which headers are supported by the response’s URL for the purposes of the CORS protocol. This variable will be used to set Access-Control-Allow-Headers in the pre-flight cors response.' + type: array + items: + type: string + maxLength: 127 + max_age: + description: 'Indicates the number of seconds (5 by default) the information provided by the `Access-Control-Allow-Methods` and `Access-Control-Allow-Headers` headers can be cached. This variable will be used to set Access-Control-Max-Age in the pre-flight cors response.' + type: integer + allow_credentials: + description: 'Indicates whether the response can be shared when the request’s credentials mode is "include". This variable will be set to Access-Control-Allow-Credentials in the rewrite action.' + type: boolean + required: [servicenames, allow_origin, allow_methods, allow_headers] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: appqoepolicies.citrix.com +spec: + group: citrix.com + names: + kind: appqoepolicy + plural: appqoepolicies + singular: appqoepolicy + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + appqoe-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to appqoe policy.' + type: array + items: + type: string + maxLength: 127 + appqoe-policy: + type: object + properties: + operation-retry: + type: object + properties: + on-reset: + description: "To set Retry on Connection Reset or Not" + type: string + enum: ['YES','NO'] + on-timeout: + description: "Time in milliseconds for retry" + type: integer + minimum: 30 + maximum: 2000 + number-of-retries: + description: "To set number of retries" + type: integer + minimum: 1 + maximum: 7 + required: [operation-retry] + appqoe-criteria: + description: 'Expression against which traffic is evaluated.' + type: string + maxLength: 1299 + direction: + description: 'Bind point to which to bind the policy.' + type: string + enum: ["REQUEST","RESPONSE"] + required: [appqoe-criteria, operation-retry] + required: [appqoe-policy] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: wildcarddnsentries.citrix.com +spec: + group: citrix.com + names: + kind: wildcarddnsentry + plural: wildcarddnsentries + singular: wildcarddnsentry + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: Current Status of the CRD + jsonPath: .status.state + - name: Message + type: string + description: Status Message + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + zone: + type: object + description: DNS configuration for a zone + properties: + domain: + type: string + description: Domain name + dnsaddrec: + type: object + description: DNS Address record + properties: + domain-ip: + type: string + description: IPv4 addresses to assign to the domain name + ttl: + type: integer + description: >- + TTL is the time for which the record must be cached + by DNS proxies + dnsaaaarec: + type: object + description: DNS AAAA record + properties: + domain-ip: + type: string + description: IPv6 addresses to assign to the domain name + ttl: + type: integer + description: >- + TTL is the time for which the record must be cached + by DNS proxies + soarec: + type: object + description: SOA record + properties: + origin-server: + type: string + description: Origin server domain + contact: + type: string + description: Admin contact + serial: + type: integer + description: >- + The secondary server uses this parameter to + determine whether it requires a zone transfer from + the primary server. + refresh: + type: integer + description: >- + Time, in seconds, for which a secondary server must + wait between successive checks on the value of the + serial number. + retry: + type: integer + description: >- + Time, in seconds, between retries if a secondary server's + attempt to contact the primary server for a zone refresh fails. + expire: + type: integer + description: >- + Time, in seconds, after which the zone data on a secondary + nameserver can no longer be considered authoritative because + all refresh and retry attempts made during the period have failed." + nsrec: + type: object + description: Name server record + properties: + nameserver: + type: string + description: Host name of the name server to add to the domain. + ttl: + type: integer + description: >- + Time to Live (TTL), in seconds, for the record. TTL + is the time for which the record must be cached by + DNS proxies. The specified TTL is applied to all the + resource records that are of the same record type + and belong to the specified domain name +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: icappolicies.citrix.com +spec: + group: citrix.com + names: + kind: icappolicy + plural: icappolicies + singular: icappolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + description: "Ingress class, if not specified then all NetScaler ingress controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + type: string + maxLength: 127 + services: + type: array + description: 'Name of the services for which the icap policy needs to be bound' + items: + type: string + icap-servers: + type: object + description: "ICAP service for the ICAP server that will be part of the load balancing setup. The service that you add provides the ICAP connection between the NetScaler appliance and load balancing virtual servers." + properties: + servers: + type: array + items: + type: object + properties: + ip: + type: string + description: 'IP of the ICAP Server' + format: ipv4 + port: + type: integer + description: 'Port number of the ICAP Server.' + minimum: 1 + maximum: 65535 + required: + - ip + - port + server-type: + type: string + description: 'Type of ICAP Server.' + enum: ['TCP', 'SSL_TCP'] + default: 'SSL_TCP' + server_host_cert: + description: |+ + 'Name of the SSL certificate to be used with ICAP server. + This certificate is mandatory for server-type SSL_TCP' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - required: [tls_secret] + - required: [preconfigured] + required: + - servers + icap: + type: array + items: + type: object + properties: + preconfigured-profile: + description: 'Names of the preconfigured ICAP profile.' + type: string + maxLength: 127 + direction: + description: 'ICAP Mode of operation. It is a mandatory argument while creating an icapprofile.' + type: string + enum: ['REQUEST','RESPONSE'] + profile: + type: object + description: 'ICAP profile(s) of the NetScaler.' + properties: + preview: + description: 'Enable or Disable preview header with ICAP request. This feature allows an ICAP server to see the beginning of a transaction, then decide if it wants to opt-out of the transaction early instead of receiving the remainder of the request message.' + type: string + enum: ["ENABLED", "DISABLED"] + preview-length: + description: 'Value of Preview Header field. NetScaler uses the minimum of this set value and the preview size received on OPTIONS' + type: integer + minimum: 0 + maximum: 4294967294 + uri: + description: 'URI representing icap service. It is a mandatory argument while creating an icapprofile.' + type: string + maxLength: 511 + host-header: + description: 'ICAP Host Header.' + type: string + maxLength: 255 + user-agent-header: + description: 'ICAP User Agent Header' + type: string + maxLength: 255 + query-params: + description: 'Query parameters to be included with ICAP request URI. Entered values should be in arg=value format. For more than one parameters, add & separated values. e.g.: arg1=val1&arg2=val2' + type: string + maxLength: 511 + connection-keep-alive: + description: 'Enable or Disable sending Allow: 204 header in ICAP request.' + type: string + enum: ["ENABLED", "DISABLED"] + insert-icap-headers: + description: 'Insert custom ICAP headers in the ICAP request to send to ICAP server. The headers can be static or can be dynamically constructed using PI Policy Expression. For example, to send static user agent and Client''s IP address, the expression can be specified as "User-Agent: NS-ICAP-Client/V1.0r0-Client-IP: "+CLIENT.IP.SRC+"r0. The NetScaler does not check the validity of the specified header name-value. You must manually validate the specified header syntax.' + type: string + maxLength: 8191 + insert-http-request: + description: 'Exact HTTP request, in the form of an expression, which the NetScaler encapsulates and sends to the ICAP server. If you set this parameter, the ICAP request is sent using only this header. This can be used when the HTTP header is not available to send or ICAP server only needs part of the incoming HTTP request. The request expression is constrained by the feature for which it is used. The NetScaler does not check the validity of this request. You must manually validate the request.' + type: string + maxLength: 8191 + req-timeout: + description: 'Time, in seconds, within which the remote server should respond to the ICAP-request. If the Netscaler does not receive full response with this time, the specified request timeout action is performed. Zero value disables this timeout functionality.' + type: integer + minimum: 0 + maximum: 86400 + req-timeout-action: + description: 'Name of the action to perform if the Vserver/Server representing the remote service does not respond with any response within the timeout value configured. The Supported actions are * BYPASS - This Ignores the remote server response and sends the request/response to Client/Server. * If the ICAP response with Encapsulated headers is not received within the request-timeout value configured, this Ignores the remote ICAP server response and sends the Full request/response to Server/Client' + type: string + enum: ['BYPASS', 'DROP', 'RESET'] + log-action: + description: 'Name of the audit message action which would be evaluated on receiving the ICAP response to emit the logs' + type: string + maxLength: 127 + required: + - uri + content-inspection-criteria: + description: 'Expression that the policy uses to determine whether to execute the specified action.' + type: string + maxLength: 1499 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). An UNDEF event indicates an internal error condition. Only the above built-in actions can be used' + type: string + maxLength: 127 + log-action: + description: 'Name of the messagelog action to use for requests that match this policy.' + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be evaluated if the current policy evaluates to TRUE.Specify one of the following values:* NEXT - Evaluate the policy with the next higher priority number.* END - End policy evaluation.Default value of goto-priority-expression: END' + type: string + operation: + description: 'Type of operation this action is going to perform. following actions are available to configure: * ICAP - forward the incoming request or response to an ICAP server for modification. * INLINEINSPECTION - forward the incoming or outgoing packets to IPS server for Intrusion Prevention. * MIRROR - Forwards cloned packets for Intrusion Detection. * NOINSPECTION - This does not forward incoming and outgoing packets to the Inspection device. * NSTRACE - capture current and further incoming packets on this transaction.' + type: string + enum: ['ICAP', 'INLINEINSPECTION', 'MIRROR', 'NOINSPECTION'] + server-failure-action: + description: 'Name of the action to perform if the Vserver representing the remote service is not UP. This is not supported for NOINSPECTION Type. The Supported actions are: * RESET - Reset the client connection by closing it. The client program, such as a browser, will handle this and may inform the user. The client may then resend the request if desired. * DROP - Drop the request without sending a response to the user. * CONTINUE - It bypasses the ContentIsnpection and Continues/resumes the Traffic-Flow to Client/Server.' + type: string + enum: ['CONTINUE', 'DROP', 'RESET'] + oneOf: + - required: [preconfigured-profile] + - required: [profile] + required: + - direction + - content-inspection-criteria + - operation + required: + - ingressclass + - services + - icap-servers + - icap +--- diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/NOTES.txt b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/NOTES.txt new file mode 100644 index 000000000..3b390d5af --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/NOTES.txt @@ -0,0 +1,14 @@ +Thank you for installing {{ .Chart.Name }}. + +Your release is named {{ .Release.Name }}. + + +To learn more about the release, try: + + $ helm status {{ .Release.Name }} + $ helm get {{ .Release.Name }} + + +To delete : + helm delete {{ .Release.Name }} + diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/_helpers.tpl b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/_helpers.tpl new file mode 100644 index 000000000..4cb1cc8f2 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/_helpers.tpl @@ -0,0 +1,93 @@ +{{- define "netscaler-cpx-ingress-controller.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 "netscaler-cpx-ingress-controller.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 -}} + +{{- define "cpxservice.fullname" -}} +{{- $name := default .Chart.Name "cpx-service" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "cpxexporter.fullname" -}} +{{- $name := default .Chart.Name "exporter" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "cpxservicemonitor.fullname" -}} +{{- $name := default .Chart.Name "netscaler-adc-cpx-servicemonitor" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "cpxservicemonitorlabel" -}} +{{- $name := default .Chart.Name "netscaler-adc-cpx-svcmon" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "cpxconfigmap.fullname" -}} +{{- $name := default .Chart.Name "cpx-nsic-configmap" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "bootupconfigmap.fullname" -}} +{{- $name := default .Chart.Name "cpx-bootup-configmap" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "netscaler-cpx-ingress-controller.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account to use +*/}} +{{- define "netscaler-cpx-ingress-controller.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "netscaler-cpx-ingress-controller.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/configmap.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/configmap.yaml new file mode 100644 index 000000000..93f29f9e3 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/configmap.yaml @@ -0,0 +1,110 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "cpxconfigmap.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + LOGLEVEL: {{ .Values.logLevel | quote | lower }} + JSONLOG: {{ .Values.jsonLog | quote | lower }} +{{- if eq (upper .Values.nsProtocol) "HTTPS" }} + NS_PROTOCOL: "https" +{{- if .Values.cpxBgpRouter }} + NS_PORT: "9443" +{{- else }} + NS_PORT: "443" +{{- end }} +{{- else }} + NS_PROTOCOL: "http" +{{- if .Values.cpxBgpRouter }} + NS_PORT: "9080" +{{- else }} + NS_PORT: "80" +{{- end }} +{{- end }} +{{- if .Values.analyticsConfig.required }} + NS_ENABLE_LABELS: {{ .Values.nsEnableLabel | quote }} +{{- end }} +{{- if ne (upper .Values.nsHTTP2ServerSide) "OFF" }} + NS_HTTP2_SERVER_SIDE: {{ .Values.nsHTTP2ServerSide | quote }} +{{- end }} +{{- if ne (toString .Values.nsCookieVersion) "0" }} + NS_COOKIE_VERSION: {{ .Values.nsCookieVersion | quote }} +{{- end }} +{{- if .Values.nsDnsNameserver }} + NS_DNS_NAMESERVER: {{ .Values.nsDnsNameserver }} +{{- end }} + +{{- if .Values.analyticsConfig.required }} + NS_ANALYTICS_CONFIG: | + distributed_tracing: + enable: {{ .Values.analyticsConfig.distributedTracing.enable | quote }} + samplingrate: {{ .Values.analyticsConfig.distributedTracing.samplingrate }} + endpoint: + {{- if not .Values.analyticsConfig.timeseries.metrics.enableNativeScrape }} + metrics: + service: {{ .Values.analyticsConfig.endpoint.metrics.service | quote }} + {{- end }} + transactions: + service: {{ .Values.analyticsConfig.endpoint.transactions.service | quote }} + timeseries: + port: {{ .Values.analyticsConfig.timeseries.port }} + metrics: + enable: {{ .Values.analyticsConfig.timeseries.metrics.enable | quote }} + mode: {{ .Values.analyticsConfig.timeseries.metrics.mode | quote }} + export_frequency: {{ .Values.analyticsConfig.timeseries.metrics.exportFrequency }} + schema_file: {{ .Values.analyticsConfig.timeseries.metrics.schemaFile | quote }} + enable_native_scrape: {{ .Values.analyticsConfig.timeseries.metrics.enableNativeScrape | quote }} + auditlogs: + enable: {{ .Values.analyticsConfig.timeseries.auditlogs.enable | quote }} + events: + enable: {{ .Values.analyticsConfig.timeseries.events.enable | quote }} + transactions: + enable: {{ .Values.analyticsConfig.transactions.enable | quote }} + port: {{ .Values.analyticsConfig.transactions.port }} +{{- end }} + +{{- if .Values.cpxBgpRouter }} +{{- if .Values.bgpSettings.required }} + NS_BGP_CONFIG: | +{{- with .Values.bgpSettings.bgpConfig }} + bgpConfig: +{{ toYaml . | indent 4 }} +{{- end }} +{{- end }} +{{- end }} + +{{- if .Values.nsLbHashAlgo.required }} + NS_LB_HASH_ALGO: | + hashFingers: {{ .Values.nsLbHashAlgo.hashFingers }} + hashAlgorithm: {{ .Values.nsLbHashAlgo.hashAlgorithm | quote }} +{{- end }} + +{{- if .Values.profileSslFrontend }} + FRONTEND_SSL_PROFILE: | + {{- toYaml .Values.profileSslFrontend | nindent 4 }} +{{- end }} + +{{- if .Values.profileTcpFrontend }} + FRONTEND_TCP_PROFILE: | + {{- toYaml .Values.profileTcpFrontend | nindent 4 }} +{{- end }} + +{{- if .Values.profileHttpFrontend }} + FRONTEND_HTTP_PROFILE: | + {{- toYaml .Values.profileHttpFrontend | nindent 4 }} +{{- end }} + +--- + +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "bootupconfigmap.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + cpx.conf: | + #NetScaler commands + {{- .Values.cpxCommands | nindent 6 -}} + #Shell commands + {{- .Values.cpxShellCommands | nindent 6 -}} + # end of file diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/deployment.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/deployment.yaml new file mode 100644 index 000000000..402d43211 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/deployment.yaml @@ -0,0 +1,479 @@ +apiVersion: apps/v1 +{{- if or .Values.cpxBgpRouter .Values.daemonSet }} +kind: DaemonSet +{{- else }} +kind: Deployment +{{- end }} +metadata: + name: {{ include "netscaler-cpx-ingress-controller.fullname" . }} + namespace: {{ .Release.Namespace }} +spec: + selector: + matchLabels: + app: {{ include "netscaler-cpx-ingress-controller.fullname" . }} +{{- if not ( or .Values.cpxBgpRouter .Values.daemonSet ) }} + replicas: {{ .Values.replicaCount }} +{{- end }} + template: + metadata: + name: {{ include "netscaler-cpx-ingress-controller.fullname" . }} + labels: + app: {{ include "netscaler-cpx-ingress-controller.fullname" . }} + adc: "citrix" + annotations: +{{- if .Values.analyticsConfig.timeseries.metrics.enableNativeScrape }} + netscaler.prometheus/scrape: "true" + netscaler.prometheus/port: {{ .Values.mgmtHttpPort | quote }} +{{- end }} +{{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} +{{- end }} + spec: + serviceAccountName: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} +{{- if .Values.cpxBgpRouter }} + hostNetwork: true +{{- end }} +{{- if .Values.hostName }} + hostname: {{ .Values.hostName }}-{{ .Release.Namespace }} +{{- end }} + containers: + - name: cpx-ingress + image: "{{ tpl .Values.image . }}" + imagePullPolicy: {{ .Values.pullPolicy }} + tty: true + securityContext: + privileged: true + env: + - name: POD_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace +{{- if .Values.cpxLicenseAggregator }} + - name: "CLA" + value: {{ .Values.cpxLicenseAggregator | quote }} +{{- else if .Values.ADMSettings.licenseServerIP }} + - name: "LS_IP" + value: {{ .Values.ADMSettings.licenseServerIP | quote }} + - name: "LS_PORT" + value: {{ .Values.ADMSettings.licenseServerPort | quote }} +{{- end }} + - name: "EULA" + value: "{{ .Values.license.accept }}" + - name: "KUBERNETES_TASK_ID" + value: "" +{{- if not .Values.cpxBgpRouter }} + - name: "MGMT_HTTP_PORT" + value: {{ .Values.mgmtHttpPort | quote }} + - name: "MGMT_HTTPS_PORT" + value: {{ .Values.mgmtHttpsPort | quote }} +{{- end }} +{{- if .Values.cpxBgpRouter }} + - name: NS_NETMODE + value: HOST +{{- if .Values.nsIP }} + - name: "NS_IP" + value: "{{ .Values.nsIP }}" +{{- end }} +{{- if .Values.nsGateway }} + - name: "NS_GATEWAY" + value: "{{ .Values.nsGateway }}" +{{- end }} +{{- end }} +{{- if .Values.ADMSettings.ADMIP }} + - name: "NS_MGMT_SERVER" + value: {{ .Values.ADMSettings.ADMIP | quote }} + - name: "NS_HTTP_PORT" + value: {{ .Values.mgmtHttpPort | quote }} + - name: "NS_HTTPS_PORT" + value: {{ .Values.mgmtHttpsPort | quote }} +{{- end }} +##Need to set env var BANDWIDTH in order to provide Bandwidth license to NetScaler CPX from ADM or CPX License Aggregator +{{- if or .Values.ADMSettings.licenseServerIP .Values.cpxLicenseAggregator }} +{{- if .Values.ADMSettings.bandWidthLicense }} + - name: "BANDWIDTH" + value: {{ .Values.ADMSettings.bandWidth | quote }} +{{- end }} +##for multiple-PE support, need to set CPX_CORES +{{- if or .Values.ADMSettings.vCPULicense .Values.ADMSettings.bandWidthLicense }} + - name: "CPX_CORES" + value: {{ .Values.ADMSettings.cpxCores | default 1 | quote }} + - name: "EDITION" + value: {{ .Values.ADMSettings.licenseEdition }} +{{- end }} +{{- if .Values.ADMSettings.platform }} + - name: "CPX_CORES" + value: {{ .Values.ADMSettings.cpxCores | default 1 | quote }} + - name: "PLATFORM" + value: "CP1000" +{{- end }} +{{- end }} +{{- if or (.Values.ADMSettings.ADMIP) (.Values.ADMSettings.licenseServerIP) }} + - name: NS_MGMT_USER + valueFrom: + secretKeyRef: + name: {{ required "Provide Secret for ADM/LicenseServer credentials" .Values.ADMSettings.loginSecret }} + key: username + - name: NS_MGMT_PASS + valueFrom: + secretKeyRef: + name: {{ required "Provide Secret for ADM/LicenseServer credentials" .Values.ADMSettings.loginSecret }} + key: password +{{- end }} +{{- if .Values.exporter.required }} + - name: "METRICS_EXPORTER_PORT" + value: {{ .Values.exporter.ports.containerPort | quote }} +{{- end }} + resources: + {{- toYaml .Values.resources | nindent 12 }} + volumeMounts: + - mountPath: /var/deviceinfo + name: shared-data + - mountPath: /cpx/ + name: cpx-volume + - mountPath: /cpx/conf + name: cpx-volume-conf + - mountPath: /cpx/bootup_conf + name: bootupconfig-volume +{{- if .Values.enableStartupProbe }} + startupProbe: + {{- toYaml .Values.startupProbe | nindent 12 }} +{{- end }} +{{- if .Values.enableLivenessProbe }} + livenessProbe: + {{- toYaml .Values.livenessProbe | nindent 12 }} +{{- end }} +{{- if .Values.nsic.required }} + # Add nsic as a sidecar + - name: nsic + image: "{{ tpl .Values.nsic.image . }}" + imagePullPolicy: {{ .Values.nsic.pullPolicy }} + env: +{{- if .Values.nsic.enableLivenessProbe }} + - name: "LIVENESS_FILE_PATH" + value: '/tmp/liveness_path.log' +{{- end }} + - name: "ENABLE_LIVENESS_PROBE" + value: {{ .Values.nsic.enableLivenessProbe | quote }} +{{- if .Values.analyticsConfig.timeseries.metrics.enableNativeScrape }} + - name: "PROM_USER" + valueFrom: + secretKeyRef: + name: {{ required "Provide Secret for read only user for native Prometheus mode" .Values.nsic.prometheusCredentialSecret }} + key: username + - name: "PROM_PASSWORD" + valueFrom: + secretKeyRef: + name: {{ required "Provide Secret for read only user for native Prometheus mode" .Values.nsic.prometheusCredentialSecret }} + key: password +{{- end }} + - name: "EULA" + value: "{{ .Values.license.accept }}" +{{- if .Values.cpxBgpRouter }} + - name: "NS_IP" + value: {{ .Values.nsIP | default "192.168.1.2" | quote }} +{{- else }} + - name: "NS_IP" + value: "127.0.0.1" +{{- end }} +{{- if .Values.rbacRole }} + - name: "SCOPE" + value: "local" +{{- end }} + - name: "NS_APPS_NAME_PREFIX" + value: {{ .Values.entityPrefix | default "k8s"| quote }} + - name: "NS_DEPLOYMENT_MODE" + value: "SIDECAR" +{{- if and .Values.openshift .Values.routeLabels }} + - name: "ROUTE_LABELS" + value: {{ .Values.routeLabels | quote}} +{{- end }} +{{- if and .Values.openshift .Values.namespaceLabels }} + - name: "NAMESPACE_LABELS" + value: {{ .Values.namespaceLabels | quote }} +{{- end }} +{{- if .Values.openshift }} + - name: "PLATFORM" + value: "OPENSHIFT" +{{- else }} + - name: "PLATFORM" + value: "KUBERNETES" +{{- end }} +{{- if .Values.cpxBgpRouter }} +{{- if eq (upper .Values.nsProtocol) "HTTPS" }} + - name: NS_PROTOCOL + value: HTTPS + - name: NS_PORT + value: "9443" +{{- else }} + - name: NS_PROTOCOL + value: HTTP + - name: NS_PORT + value: "9080" +{{- end }} +{{- if .Values.bgpPort }} + - name: "BGP_PORT" + value: {{ .Values.bgpPort | quote }} +{{- end }} +{{- end }} + - name: "NS_ENABLE_MONITORING" + value: "YES" +{{- if .Values.logProxy }} + - name: "NS_LOGPROXY" + value: {{ .Values.logProxy | quote }} +{{- end }} +{{- if .Values.ingressIP }} + - name: "NS_VIP" + value: {{ .Values.ingressIP | quote }} +{{- end }} +{{- if .Values.nitroReadTimeout }} + - name: "NS_NITRO_READ_TIMEOUT" + value: "{{ .Values.nitroReadTimeout }}" +{{- end }} + - name: POD_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: NODE_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName +{{- if .Values.kubernetesURL }} + - name: "kubernetes_url" + value: "{{ .Values.kubernetesURL }}" +{{- end }} +{{- if .Values.disableOpenshiftRoutes }} + - name: "DISABLE_OPENSHIFT_ROUTES" + value: "{{ .Values.disableOpenshiftRoutes }}" +{{- end }} +{{- if .Values.nsConfigDnsRec }} + - name: "NS_CONFIG_DNS_REC" + value: "{{ .Values.nsConfigDnsRec }}" +{{- end }} +{{- if .Values.nsSvcLbDnsRec }} + - name: "NS_SVC_LB_DNS_REC" + value: "{{ .Values.nsSvcLbDnsRec }}" +{{- end }} +{{- if .Values.optimizeEndpointBinding }} + - name: "OPTIMIZE_ENDPOINT_BINDING" + value: "{{ .Values.optimizeEndpointBinding }}" +{{- end }} +{{- if .Values.cpxBgpRouter }} + securityContext: + runAsUser: 0 + capabilities: + add: + - NET_ADMIN +{{- end }} + args: + - --configmap + {{ .Release.Namespace }}/{{ include "cpxconfigmap.fullname" . }} +{{- if .Values.ipam }} + - --ipam + citrix-ipam-controller +{{- end }} +{{- if .Values.disableAPIServerCertVerify }} + - --disable-apiserver-cert-verify + {{ .Values.disableAPIServerCertVerify }} +{{- end }} +{{- if .Values.cpxBgpRouter }} + - --deployment-type + kube-bgp-router +{{- end }} +{{- if .Values.ingressClass }} + - --ingress-classes +{{- range .Values.ingressClass}} + {{.}} +{{- end }} +{{- end }} +{{- if .Values.defaultSSLCertSecret }} + - --default-ssl-certificate + {{ .Release.Namespace }}/{{ .Values.defaultSSLCertSecret }} +{{- end }} +{{- if .Values.defaultSSLSNICertSecret }} + - --default-ssl-sni-certificate + {{ .Release.Namespace }}/{{ .Values.defaultSSLSNICertSecret }} +{{- end }} +{{- if .Values.updateIngressStatus }} + - --update-ingress-status + yes +{{- end }} + volumeMounts: + - mountPath: /var/deviceinfo + name: shared-data +{{- if .Values.nsic.enableReadinessProbe }} + readinessProbe: + {{- toYaml .Values.nsic.readinessProbe | nindent 12 }} +{{- end }} +{{- if .Values.nsic.enableLivenessProbe }} + livenessProbe: + {{- toYaml .Values.nsic.livenessProbe | nindent 12 }} +{{- end }} + resources: + {{- toYaml .Values.nsic.resources | nindent 12 }} +{{- end }} +{{- if .Values.exporter.required }} + - name: exporter + image: "{{ tpl .Values.exporter.image . }}" + imagePullPolicy: {{ .Values.exporter.pullPolicy }} + args: + - "--secure=no" +{{- if .Values.cpxBgpRouter }} + - --target-nsip={{ .Values.nsIP | default "192.168.1.2" }}:9080 +{{- else }} + - "--target-nsip=127.0.0.1" +{{- end }} + - "--port={{ .Values.exporter.ports.containerPort }}" + env: + - name: "NS_DEPLOYMENT_MODE" + value: "SIDECAR" + securityContext: + readOnlyRootFilesystem: true + volumeMounts: + - mountPath: /var/deviceinfo + name: shared-data + resources: + {{- toYaml .Values.exporter.resources | nindent 12 }} +{{- end }} + volumes: + - name: shared-data + emptyDir: {} + - name: cpx-volume + emptyDir: {} + - name: cpx-volume-conf + emptyDir: {} + - name: bootupconfig-volume + configMap: + name: {{ include "bootupconfigmap.fullname" . }} +{{- if and .Values.nodeSelector.key .Values.nodeSelector.value }} + nodeSelector: + {{ .Values.nodeSelector.key }}: {{ .Values.nodeSelector.value }} +{{- end }} +{{- if .Values.tolerations }} + tolerations: {{ .Values.tolerations | toYaml | nindent 8 }} +{{- end }} +{{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} +{{- end }} + +--- +{{- if .Values.cpxBgpRouter }} +{{- if .Values.exporter.required }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "cpxexporter.fullname" . }} + labels: + app: {{ include "cpxexporter.fullname" . }} + service-type: {{ include "cpxservicemonitorlabel" . }} +spec: + type: ClusterIP + ports: + - port: {{ .Values.exporter.ports.containerPort }} + targetPort: {{ .Values.exporter.ports.containerPort }} + name: exporter-port + selector: + app: {{ include "netscaler-cpx-ingress-controller.fullname" . }} +{{- end }} +{{- else }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "cpxservice.fullname" . }} + labels: + app: cpx-service + service-type: {{ include "cpxservicemonitorlabel" . }} + cpx: {{ include "netscaler-cpx-ingress-controller.fullname" . }} +{{- if .Values.serviceAnnotations }} + annotations: +{{- with .Values.serviceAnnotations }} +{{ toYaml . | indent 4 }} +{{- end }} +{{- end }} +spec: +{{- if or .Values.serviceType.loadBalancer.enabled .Values.serviceType.nodePort.enabled }} + externalTrafficPolicy: {{ .Values.serviceSpec.externalTrafficPolicy }} +{{- end }} +{{- if .Values.serviceType.loadBalancer.enabled }} + type: LoadBalancer +{{- if .Values.serviceSpec.loadBalancerIP }} + loadBalancerIP: {{ .Values.serviceSpec.loadBalancerIP }} +{{- end }} +{{- else if .Values.serviceType.nodePort.enabled }} + type: NodePort +{{- end }} +{{- if and .Values.serviceType.loadBalancer.enabled .Values.serviceSpec.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{- range .Values.serviceSpec.loadBalancerSourceRanges}} + - {{.}} +{{- end }} +{{- end }} + ports: +{{- if .Values.servicePorts }} +{{- with .Values.servicePorts }} +{{ toYaml . | indent 2 }} +{{- end }} +{{- else }} + - port: 80 + protocol: TCP + name: http +{{- if and .Values.serviceType.nodePort.enabled .Values.serviceType.nodePort.httpPort }} + nodePort: {{ .Values.serviceType.nodePort.httpPort }} +{{- end }} + - port: 443 + protocol: TCP + name: https +{{- if and .Values.serviceType.nodePort.enabled .Values.serviceType.nodePort.httpsPort }} + nodePort: {{ .Values.serviceType.nodePort.httpsPort}} +{{- end }} +{{- end }} +{{- if .Values.exporter.required }} + - port: {{ .Values.exporter.ports.containerPort }} + targetPort: {{ .Values.exporter.ports.containerPort }} + name: exporter-port +{{- end }} + selector: + app: {{ include "netscaler-cpx-ingress-controller.fullname" . }} +{{- end }} + +--- + +{{- if .Values.exporter.required }} + +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "cpxservicemonitor.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + servicemonitor: netscaler-cpx + {{- with .Values.exporter.serviceMonitorExtraLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + endpoints: + - interval: 30s + port: exporter-port + selector: + matchLabels: + service-type: {{ include "cpxservicemonitorlabel" . }} + namespaceSelector: + matchNames: + - monitoring + - default + - {{ .Release.Namespace }} + +{{- end }} diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/ingressclass.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/ingressclass.yaml new file mode 100644 index 000000000..da86715cc --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/ingressclass.yaml @@ -0,0 +1,18 @@ +{{- $default := .Values.setAsDefaultIngressClass -}} +{{- if semverCompare ">=1.19.0-0" .Capabilities.KubeVersion.GitVersion }} +{{- if .Values.ingressClass }} +{{- range .Values.ingressClass }} +apiVersion: networking.k8s.io/v1 +kind: IngressClass +metadata: + name: {{ . | quote }} +{{- if $default }} + annotations: + ingressclass.kubernetes.io/is-default-class: "true" +{{- end }} +spec: + controller: citrix.com/ingress-controller +--- +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/rbac.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/rbac.yaml new file mode 100644 index 000000000..103d8e168 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/templates/rbac.yaml @@ -0,0 +1,106 @@ +{{- if not .Values.rbacRole }} +kind: ClusterRole +{{- else }} +kind: Role +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} +{{- if .Values.rbacRole }} + namespace: {{ .Release.Namespace }} +{{- end }} +rules: + - apiGroups: [""] +{{- if .Values.openshift }} + resources: ["endpoints", "pods", "secrets", "routes", "tokenreviews", "subjectaccessreviews", "nodes", "namespaces", "configmaps", "services"] +{{- else }} + resources: ["endpoints", "pods", "secrets", "routes", "nodes", "namespaces", "configmaps", "services"] +{{- end }} + verbs: ["get", "list", "watch"] + # services/status is needed to update the loadbalancer IP in service status for integrating + # service of type LoadBalancer with external-dns + - apiGroups: [""] + resources: ["services/status"] + verbs: ["patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create"] + - apiGroups: ["extensions", "networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["extensions","networking.k8s.io"] + resources: ["ingresses/status"] + verbs: ["patch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingressclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["get", "list", "watch"] + - apiGroups: ["apps"] + resources: ["deployments"] + verbs: ["get", "list", "watch"] + - apiGroups: ["citrix.com"] + resources: ["rewritepolicies", "icappolicies", "continuousdeployments", "authpolicies", "ratelimits", "listeners", "httproutes", "wafs", "apigatewaypolicies", "bots", "corspolicies", "appqoepolicies", "wildcarddnsentries"] + verbs: ["get", "list", "watch", "create", "delete", "patch"] + - apiGroups: ["citrix.com"] + resources: ["rewritepolicies/status", "icappolicies/status", "continuousdeployments/status", "authpolicies/status", "ratelimits/status", "listeners/status", "httproutes/status", "wafs/status", "apigatewaypolicies/status", "bots/status", "corspolicies/status", "appqoepolicies/status", "wildcarddnsentries/status"] + verbs: ["patch"] + - apiGroups: ["citrix.com"] + resources: ["vips"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: ["crd.projectcalico.org"] + resources: ["ipamblocks"] + verbs: ["get", "list", "watch"] +{{- if .Values.openshift }} + - apiGroups: ["route.openshift.io"] + resources: ["routes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["network.openshift.io"] + resources: ["hostsubnets"] + verbs: ["get", "list", "watch"] + - apiGroups: ["config.openshift.io"] + resources: ["networks"] + verbs: ["get", "list"] +{{- end }} + +--- + +{{- if not .Values.rbacRole }} +kind: ClusterRoleBinding +{{- else }} +kind: RoleBinding +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} +{{- if .Values.rbacRole }} + namespace: {{ .Release.Namespace }} +{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io +{{- if not .Values.rbacRole }} + kind: ClusterRole +{{- else }} + kind: Role +{{- end }} + name: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} +subjects: +- kind: ServiceAccount + name: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + +--- + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "netscaler-cpx-ingress-controller.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- if .Values.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.imagePullSecrets }} +- name: {{.}} +{{- end }} +{{- end }} + diff --git a/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/values.yaml b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/values.yaml new file mode 100644 index 000000000..16cf34e48 --- /dev/null +++ b/charts/netscaler/netscaler-cpx-with-ingress-controller/2.2.10/values.yaml @@ -0,0 +1,306 @@ +# Default values for netscaler-cpx-with-ingress-controller. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# NetScaler CPX config details +imageRegistry: quay.io +imageRepository: netscaler/netscaler-cpx +imageTag: 14.1-25.111 +image: "{{ .Values.imageRegistry }}/{{ .Values.imageRepository }}:{{ .Values.imageTag }}" +pullPolicy: IfNotPresent +imagePullSecrets: [] +daemonSet: False +nameOverride: "" +replicaCount: 1 +fullnameOverride: "" +mgmtHttpPort: 9080 +mgmtHttpsPort: 9443 +openshift: false +nsHTTP2ServerSide: "OFF" +nsCookieVersion: "0" +nsConfigDnsRec: false +nsSvcLbDnsRec: false +nsDnsNameserver: "" +nsEnableLabel: true +optimizeEndpointBinding: false +routeLabels: "" +namespaceLabels: "" +hostName: "" + +# Service Type LoadBalancer and ingress support with CPX through BGP advertisement +# If you enable this, CPX is run as DaemonSet. Please edit the bgpSettings for configuring +# BGP neighbors for propgation of external IPs. +cpxBgpRouter: false + +# If cpxBgpRouter is true, then this is the NSIP used by CPX for internal communication +nsIP: 192.168.1.2 + +# If cpxBgpRouter is true, then this is the Gateway used by CPX for internal communication +nsGateway: 192.168.1.1 + +# Protocol used for communication between NetScaler Ingress Controller sidecar and NetScaler CPX +nsProtocol: http + +# External IP for ingress resource when bgpRouter is set to True +ingressIP: "" + +# If IPAM controller is used for auto allocation of the external IP for service of type LoadBalancer, set this option to true +ipam: False + +# Enable RBAC role (so called local role), by default NSIC deployed with ClusterRole. +# below variable to deploy NSIC with RBAC role, only ingress service supported with this config +rbacRole: False + +# API server Cert verification can be disabled, while communicating with API Server, if disableAPIServerCertVerify set to True +disableAPIServerCertVerify: False + +cpxLicenseAggregator: "" + +nodeSelector: + key: "" + value: "" +tolerations: [] + +serviceType: + loadBalancer: + enabled: False + nodePort: + enabled: False + httpPort: "" + httpsPort: "" + +serviceAnnotations: {} + +serviceSpec: + externalTrafficPolicy: "Cluster" + loadBalancerIP: "" + loadBalancerSourceRanges: [] + +servicePorts: [] + +# NetScaler Ingress Controller config details +nsic: + imageRegistry: quay.io + imageRepository: netscaler/netscaler-k8s-ingress-controller + imageTag: 2.2.10 + image: "{{ .Values.nsic.imageRegistry }}/{{ .Values.nsic.imageRepository }}:{{ .Values.nsic.imageTag }}" + pullPolicy: IfNotPresent + required: true + resources: + requests: + cpu: 32m + memory: 128Mi + # Following values depends on no of ingresses configured by Ingress Controllers, so it is + # advised to test with maximum no of ingresses to set these values. + # limits: + # cpu: 1000m + # memory: 1000Mi + limits: {} + # Following values depends on no of ingresses configured by Ingress Controllers, so it is + # advised to test with maximum no of ingresses to set these values. + # limits: + # cpu: 1000m + # memory: 1000Mi + prometheusCredentialSecret: "" # K8s Secret Name for read only user creation for native Prometheus support + enableLivenessProbe: True + livenessProbe: + exec: + command: + - /bin/sh + - -c + - | + FILE_PATH="$LIVENESS_FILE_PATH" + [ -f "$FILE_PATH" ] && [ $(( $(date +%s) - $(stat -c %Y "$FILE_PATH") )) -lt 60 ] && exit 0 || exit 1 + initialDelaySeconds: 30 + periodSeconds: 60 + + enableReadinessProbe: True + readinessProbe: + exec: + command: + - cat + - /tmp/readiness + initialDelaySeconds: 50 + periodSeconds: 60 + failureThreshold: 3 + successThreshold: 1 + +entityPrefix: '' +license: + accept: no +ingressClass: [] +setAsDefaultIngressClass: False +# nitroReadTimeout is timeout value in seconds for nitro api read timeout(default is 20) +nitroReadTimeout: 20 +logLevel: INFO +jsonLog: false +defaultSSLCertSecret: "" +defaultSSLSNICertSecret: "" +updateIngressStatus: False +logProxy: "" +kubernetesURL: "" +disableOpenshiftRoutes: false +profileSslFrontend: {} + # preconfigured: my_ssl_profile + # OR + # config: + # tls13: 'ENABLED' + # hsts: 'ENABLED' +profileHttpFrontend: {} + # preconfigured: my_http_profile + # OR + # config: + # dropinvalreqs: 'ENABLED' + # websocket: 'ENABLED' +profileTcpFrontend: {} + # preconfigured: my_tcp_profile + # OR + # config: + # sack: 'ENABLED' + # nagle: 'ENABLED' + + +# NetScaler ADM/License Server config details +ADMSettings: + licenseServerIP: + licenseServerPort: 27000 + ADMIP: "" + loginSecret: "" + bandWidthLicense: false + bandWidth: 1000 #bandwidth value shoule be in Mbps + vCPULicense: false + cpxCores: 1 + platform: false + licenseEdition: PLATINUM + +# Exporter config details +exporter: + required: false + imageRegistry: quay.io + imageRepository: netscaler/netscaler-adc-metrics-exporter + imageTag: 1.4.9 + image: "{{ .Values.exporter.imageRegistry }}/{{ .Values.exporter.imageRepository }}:{{ .Values.exporter.imageTag }}" + pullPolicy: IfNotPresent + ports: + containerPort: 8888 + 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 + serviceMonitorExtraLabels: {} + +# Config required to be done by NetScaler Ingress Controller for sending metrics to NetScaler Observability Exporter +analyticsConfig: + required: false + distributedTracing: + enable: false + samplingrate: 100 + endpoint: + metrics: + service: "" + transactions: + service: "" + timeseries: + port: 5563 + metrics: + enable: false + mode: 'avro' + exportFrequency: 30 + schemaFile: schema.json + enableNativeScrape: false + auditlogs: + enable: false + events: + enable: false + transactions: + enable: false + port: 5557 + +# BGP configurations: local AS, remote AS and remote address is mandatory to provide. Please do the approrpiate changes with respect to your environment +bgpSettings: + # When bgpConfig is configured correctly, set the required to true for the configuration to be applied. + required: false + bgpConfig: + - bgpRouter: + # Local AS number for BGP advertisement + localAS: 100 + neighbor: + # Address of the nighbor router for BGP advertisement + - address: "" + # Remote AS number + remoteAS: 100 + advertisementInterval: 10 + ASOriginationInterval: 10 + +bgpPort: 179 + +nsLbHashAlgo: + required: false + hashFingers: 256 + hashAlgorithm: 'DEFAULT' + +# Specifies whether a ServiceAccount should be created +serviceAccount: + 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: + +podAnnotations: {} + +# This is the resource for CPX container. +resources: + requests: + cpu: 128m + memory: 500Mi + limits: {} + # limits: + # cpu: 500m + # memory: 512Mi + +affinity: {} + +enableStartupProbe: True +startupProbe: + initialDelaySeconds: 30 + periodSeconds: 5 + failureThreshold: 20 + successThreshold: 1 + exec: + command: + - /bin/ping + - -c 1 + - 192.0.0.1 + +enableLivenessProbe: True +livenessProbe: + exec: + command: + - ls + - /tmp/cpx_started + periodSeconds: 5 + failureThreshold: 3 + successThreshold: 1 + + + +# cpxCommands: to provide global config to be applied in CPX. The commands will be executed in order. For e.g. +# add rewrite action rw_act_x_forwarded_proto insert_http_header X-Forwarded-Proto "\"https\"" +# add rewrite policy rw_pol_x_forwarded_proto CLIENT.SSL.IS_SSL rw_act_x_forwarded_proto +# bind rewrite global rw_pol_x_forwarded_proto 10 -type REQ_OVERRIDE +cpxCommands: | + + +# cpxShellCommands: to provide commands that need to be executed in shell of CPX. For e.g. +# touch /etc/a.txt +# echo "this is a" > /etc/a.txt +# echo "this is the file" >> /etc/a.txt +# ls >> /etc/a.txt +cpxShellCommands: | diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/Chart.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/Chart.yaml new file mode 100644 index 000000000..d92ddf3fd --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/Chart.yaml @@ -0,0 +1,20 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: NetScaler Ingress Controller + catalog.cattle.io/kube-version: '>=v1.16.0-0' + catalog.cattle.io/release-name: netscaler-ingress-controller +apiVersion: v2 +appVersion: 2.2.10 +description: A Helm chart for NetScaler Ingress Controller configuring MPX/VPX. +home: https://www.netscaler.com +icon: file://assets/icons/netscaler-ingress-controller.png +kubeVersion: '>=v1.16.0-0' +maintainers: +- email: priyanka.sharma@cloud.com + name: priyankash-citrix +- email: subash.dangol@cloud.com + name: subashd +name: netscaler-ingress-controller +sources: +- https://github.com/netscaler/netscaler-k8s-ingress-controller +version: 2.2.10 diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/README.md b/charts/netscaler/netscaler-ingress-controller/2.2.10/README.md new file mode 100644 index 000000000..6454196db --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/README.md @@ -0,0 +1,533 @@ +# NetScaler Ingress Controller + +[NetScaler](https://www.netscaler.com/) provides an Ingress Controller for NetScaler MPX (hardware), NetScaler VPX (virtualized), and [NetScaler CPX](https://docs.netscaler.com/en-us/citrix-adc-cpx//13/about.html) (containerized) for [bare metal](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/deployment/baremetal) and [cloud](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/deployment) deployments. It configures one (or more) NetScaler based on the Ingress resource configuration in [Kubernetes](https://kubernetes.io/) or in [OpenShift](https://www.openshift.com) cluster. + +## TL;DR; + +### For Kubernetes + ``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + + helm install nsic netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret= + ``` + +### For OpenShift + + ``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + + helm install nsic netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,openshift=true + ``` + +> **Important:** +> +> The `license.accept` argument is mandatory. Ensure that you set the value as `yes` to accept the terms and conditions of the NetScaler license. + +> **NOTE:** +> +> The CRDs supported by NetScaler will be installed automatically with the installation of the Helm Charts if CRDs are not already available in the cluster. + +## Introduction +This Helm chart deploys NetScaler ingress controller in the [Kubernetes](https://kubernetes.io) or in the [Openshift](https://www.openshift.com) cluster using [Helm](https://helm.sh) package manager. + +### Prerequisites + +- The [Kubernetes](https://kubernetes.io/) version should be 1.16 and above if using Kubernetes environment. +- The [Openshift](https://www.openshift.com) version 4.8 or later if using OpenShift platform. +- The [Helm](https://helm.sh/) version 3.x or later. You can follow instruction given [here](https://github.com/netscaler/netscaler-helm-charts/blob/master/Helm_Installation_version_3.md) to install the same. +- You determine the NS_IP IP address needed by the controller to communicate with NetScaler. The IP address might be anyone of the following depending on the type of NetScaler deployment: + + - (Standalone appliances) NSIP - The management IP address of a standalone NetScaler appliance. For more information, see [IP Addressing in NetScaler](https://docs.netscaler.com/en-us/citrix-adc/current-release/networking/ip-addressing.html). + + - (Appliances in High Availability mode) SNIP - The subnet IP address. For more information, see [IP Addressing in NetScaler](https://docs.netscaler.com/en-us/citrix-adc/current-release/networking/ip-addressing.html). + + - (Appliances in Clustered mode) CLIP - The cluster management IP (CLIP) address for a clustered NetScaler deployment. For more information, see [IP addressing for a cluster](https://docs.netscaler.com/en-us/citrix-adc/current-release/clustering/cluster-overview/ip-addressing.html). + +- You have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator), if you want to view the metrics of the NetScaler CPX collected by the [metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics). + +- The user name and password of the NetScaler VPX or MPX appliance used as the ingress device. The NetScaler appliance needs to have system user account (non-default) with certain privileges so that NetScaler ingress controller can configure the NetScaler VPX or MPX appliance. For instructions to create the system user account on NetScaler, see [Create System User Account for NSIC in NetScaler](#create-system-user-account-for-nsic-in-citrix-adc). + + You can pass user name and password using Kubernetes secrets. Create a Kubernetes secret for the user name and password using the following command: + + ``` + kubectl create secret generic nslogin --from-literal=username='nsic' --from-literal=password='mypassword' + ``` + +#### Create system User account for NetScaler ingress controller in NetScaler + +NetScaler ingress controller configures the NetScaler using a system user account of the NetScaler. The system user account should have certain privileges so that the NSIC has permission configure the following on the NetScaler: + +- Add, Delete, or View Content Switching (CS) virtual server +- Configure CS policies and actions +- Configure Load Balancing (LB) virtual server +- Configure Service groups +- Cofigure SSl certkeys +- Configure routes +- Configure user monitors +- Add system file (for uploading SSL certkeys from Kubernetes) +- Configure Virtual IP address (VIP) +- Check the status of the NetScaler appliance + +> **Note:** +> +> The system user account would have privileges based on the command policy that you define. + +To create the system user account, do the following: + +1. Log on to the NetScaler appliance. Perform the following: + 1. Use an SSH client, such as PuTTy, to open an SSH connection to the NetScaler appliance. + + 2. Log on to the appliance by using the administrator credentials. + +2. Create the system user account using the following command: + + ``` + add system user + ``` + + For example: + + ``` + add system user nsic mypassword + ``` + +3. Create a policy to provide required permissions to the system user account. Use the following command: + + ``` + add cmdpolicy nsic-policy ALLOW '^(\?!shell)(\?!sftp)(\?!scp)(\?!batch)(\?!source)(\?!.*superuser)(\?!.*nsroot)(\?!install)(\?!show\s+system\s+(user|cmdPolicy|file))(\?!(set|add|rm|create|export|kill)\s+system)(\?!(unbind|bind)\s+system\s+(user|group))(\?!diff\s+ns\s+config)(\?!(set|unset|add|rm|bind|unbind|switch)\s+ns\s+partition).*|(^install\s*(wi|wf))|(^\S+\s+system\s+file)^(\?!shell)(\?!sftp)(\?!scp)(\?!batch)(\?!source)(\?!.*superuser)(\?!.*nsroot)(\?!install)(\?!show\s+system\s+(user|cmdPolicy|file))(\?!(set|add|rm|create|export|kill)\s+system)(\?!(unbind|bind)\s+system\s+(user|group))(\?!diff\s+ns\s+config)(\?!(set|unset|add|rm|bind|unbind|switch)\s+ns\s+partition).*|(^install\s*(wi|wf))|(^\S+\s+system\s+file)' + ``` + + **Note**: The system user account would have privileges based on the command policy that you define. + The command policy mentioned in ***step 3*** is similar to the built-in `sysAdmin` command policy with another permission to upload files. + + The command policy spec provided above have already escaped special characters for easier copy pasting into the NetScaler command line. + + For configuring the command policy from NetScaler Configuration Wizard (GUI), use the below command policy spec. + + ``` + ^(?!shell)(?!sftp)(?!scp)(?!batch)(?!source)(?!.*superuser)(?!.*nsroot)(?!install)(?!show\s+system\s+(user|cmdPolicy|file))(?!(set|add|rm|create|export|kill)\s+system)(?!(unbind|bind)\s+system\s+(user|group))(?!diff\s+ns\s+config)(?!(set|unset|add|rm|bind|unbind|switch)\s+ns\s+partition).*|(^install\s*(wi|wf))|(^\S+\s+system\s+file)^(?!shell)(?!sftp)(?!scp)(?!batch)(?!source)(?!.*superuser)(?!.*nsroot)(?!install)(?!show\s+system\s+(user|cmdPolicy|file))(?!(set|add|rm|create|export|kill)\s+system)(?!(unbind|bind)\s+system\s+(user|group))(?!diff\s+ns\s+config)(?!(set|unset|add|rm|bind|unbind|switch)\s+ns\s+partition).*|(^install\s*(wi|wf))|(^\S+\s+system\s+file) + ``` + +4. Bind the policy to the system user account using the following command: + + ``` + bind system user nsic nsic-policy 0 + ``` + +## Installing the Chart +Add the NetScaler Ingress Controller helm chart repository using command: + +``` + helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ +``` + +### For Kubernetes: +#### 1. NetScaler Ingress Controller +To install the chart with the release name, `my-release`, use the following command: + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,ingressClass[0]= + ``` + +> **Note:** +> +> By default the chart installs the recommended [RBAC](https://kubernetes.io/docs/admin/authorization/rbac/) roles and role bindings. + +The command deploys NetScaler ingress controller on Kubernetes cluster with the default configuration. The [configuration](#configuration) section lists the mandatory and optional parameters that you can configure during installation. + +#### 2. NetScaler Ingress Controller with Exporter +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed along with NetScaler ingress controller and collects metrics from the NetScaler instances. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. + +> **Note:** +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator). + +Use the following command for this: + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,ingressClass[0]=,exporter.required=true + ``` + +### For Openshift: +Add the name of the service account created when the chart is deployed to the privileged Security Context Constraints of OpenShift: + + ``` + oc adm policy add-scc-to-user privileged system:serviceaccount:: + ``` + +#### 1. NetScaler Ingress Controller +To install the chart with the release name, `my-release`, use the following command: + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,openshift=true + ``` + +The command deploys NetScaler ingress controller on your Openshift cluster in the default configuration. The [configuration](#configuration) section lists the mandatory and optional parameters that you can configure during installation. + +#### 2. NetScaler Ingress Controller with Exporter +[Metrics exporter](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/metrics-visualizer#visualization-of-metrics) can be deployed along with NetScaler ingress controller and collects metrics from the NetScaler instances. You can then [visualize these metrics](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/metrics/promotheus-grafana.html) using Prometheus Operator and Grafana. + +> **Note:** +> Ensure that you have installed [Prometheus Operator](https://github.com/coreos/prometheus-operator) + +Use the following command for this: + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,openshift=true,exporter.required=true + ``` + +### Installed components + +The following components are installed: + +- [NetScaler ingress controller](https://github.com/netscaler/netscaler-k8s-ingress-controller) +- [Exporter](https://github.com/netscaler/netscaler-adc-metrics-exporter) (if enabled) + +## Configuration for ServiceGraph: + If NetScaler VPX/MPX need to send data to the NetScaler ADM to bring up the servicegraph, then the below steps can be followed to install NetScaler ingress controller for NetScaler VPX/MPX. NetScaler ingress controller configures NetScaler VPX/MPX with the configuration required for servicegraph. + + 1. Create secret using NetScaler VPX credentials, which will be used by NetScaler ingress controller for configuring NetScaler VPX/MPX: + + kubectl create secret generic nslogin --from-literal=username='nsic' --from-literal=password='mypassword' + + 2. Deploy NetScaler ingress controller using helm command: + + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,nsVIP=,license.accept=yes,adcCredentialSecret=,analyticsConfig.required=true,analyticsConfig.timeseries.metrics.enable=true,analyticsConfig.timeseries.port=5563,analyticsConfig.distributedTracing.enable=true,analyticsConfig.transactions.enable=true,analyticsConfig.transactions.port=5557,analyticsConfig.endpoint.metrics.service= + +> **Note:** +> If container agent is being used here for NetScaler ADM, please provide `podIP` of container agent in the `analyticsConfig.endpoint.metrics.service` parameter. + +## CRDs configuration + +CRDs will be installed when we install NetScaler ingress controller via Helm automatically if CRDs are not installed in cluster already. If you wish to skip the CRD installation step, you can pass the --skip-crds flag. For more information about this option in Helm please see [this](https://helm.sh/docs/chart_best_practices/custom_resource_definitions/). + +There are a few examples of how to use these CRDs, which are placed in the folder: [Example-CRDs](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds). Refer to them and install as needed, using the following command: +```kubectl create -f ``` + +### Details of the supported CRDs: + +#### authpolicies CRD: + +Authentication policies are used to enforce access restrictions to resources hosted by an application or an API server. + +NetScaler provides a Kubernetes CustomResourceDefinitions (CRDs) called the [Auth CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/auth) that you can use with the NetScaler ingress controller to define authentication policies on the ingress NetScaler. + +Example file: [auth_example.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/auth_example.yaml) + +#### continuousdeployments CRD for canary: + +Canary release is a technique to reduce the risk of introducing a new software version in production by first rolling out the change to a small subset of users. After user validation, the application is rolled out to the larger set of users. NetScaler-Integrated [Canary Deployment solution](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/canary) stitches together all components of continuous delivery (CD) and makes canary deployment easier for the application developers. + +#### httproutes and listeners CRDs for contentrouting: + +[Content Routing (CR)](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/contentrouting) is the execution of defined rules that determine the placement and configuration of network traffic between users and web applications, based on the content being sent. For example, a pattern in the URL or header fields of the request. + +Example files: [HTTPRoute_crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/HTTPRoute_crd.yaml), [Listener_crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/Listener_crd.yaml) + +#### ratelimits CRD: + +In a Kubernetes deployment, you can [rate limit the requests](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/ratelimit) to the resources on the back end server or services using rate limiting feature provided by the ingress NetScaler. + +Example files: [ratelimit-example1.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/ratelimit-example1.yaml), [ratelimit-example2.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/ratelimit-example2.yaml) + +#### vips CRD: + +NetScaler provides a CustomResourceDefinitions (CRD) called [VIP](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/crd/vip) for asynchronous communication between the IPAM controller and NetScaler ingress controller. + +The IPAM controller is provided by NetScaler for IP address management. It allocates IP address to the service from a defined IP address range. The NetScaler ingress controller configures the IP address allocated to the service as virtual IP (VIP) in NetScaler ADX VPX. And, the service is exposed using the IP address. + +When a new service is created, the NetScaler ingress controller creates a CRD object for the service with an empty IP address field. The IPAM Controller listens to addition, deletion, or modification of the CRD and updates it with an IP address to the CRD. Once the CRD object is updated, the NetScaler ingress controller automatically configures NetScaler-specfic configuration in the tier-1 NetScaler VPX. + +#### rewritepolicies CRD: + +In kubernetes environment, to deploy specific layer 7 policies to handle scenarios such as, redirecting HTTP traffic to a specific URL, blocking a set of IP addresses to mitigate DDoS attacks, imposing HTTP to HTTPS and so on, requires you to add appropriate libraries within the microservices and manually configure the policies. Instead, you can use the [Rewrite and Responder features](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/crd/rewrite-policy/rewrite-responder-policies-deployment.yaml) provided by the Ingress NetScaler device to deploy these policies. + +Example files: [target-url-rewrite.yaml](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/simplified-deployment-usecases/CRDs/rewrite.md#url-manipulation) + +#### wafs CRD: + +[WAF CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/waf.md) can be used to configure the web application firewall policies with the NetScaler ingress controller on the NetScaler VPX, MPX, SDX, and CPX. The WAF CRD enables communication between the NetScaler ingress controller and NetScaler for enforcing web application firewall policies. + +In a Kubernetes deployment, you can enforce a web application firewall policy to protect the server using the WAF CRD. For more information about web application firewall, see [Web application security](https://docs.netscaler.com/en-us/citrix-adc/current-release/application-firewall/introduction-to-citrix-web-app-firewall.html). + +Example files: [wafhtmlxsssql.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/wafhtmlxsssql.yaml) + +#### apigateway CRD: + +API Gateway CRD is used to configure gitops framework on NetScaler API gateway. This solution enables NetScaler ingress controller to generate API gateway configurations out of Open API Specification documents checked in to git repository by API developers and designers. + +Example files: [api-gateway-crd-instance.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/api-gateway-crd-instance.yaml) +#### bots CRD: + +[BOT CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/bot.md) You can use Bot CRDs to configure the bot management policies with the NetScaler ingress controller on the NetScaler VPX. The Bot custom resource definition enables communication between the NetScaler ingress controller and NetScaler for enforcing bot management policies. + +In a Kubernetes deployment, you can enforce bot management policy on therequests and responses from and to the server using the Bot CRDs. For more information on security vulnerabilities, see [Bot Detection](https://docs.netscaler.com/en-us/citrix-adc/current-release/bot-management/bot-detection.html). + +Example files: [botallowlist.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/botallowlist.yaml) + +#### CORS CRD: + +[CORS CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/cors.md) Cross-origin resource sharing (CORS) is a mechanism allows a web application running under one domain to securely access resources in another domain. You can configure CORS policies on NetScaler using NetScaler ingress controller to allow one domain (the origin domain) to call APIs in another domain. For more information, see the [cross-origin resource sharing CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/cors.md) documentation. + +Example files: [cors-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/corspolicy-example.yaml) + +#### APPQOE CRD: + +[APPQOE CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/appqoe.md) When a NetScaler appliance receives an HTTP request and forwards it to a back-end server, sometimes there may be connection failures with the back-end server. You can configure the request-retry feature on NetScaler to forward the request to the next available server, instead of sending the reset to the client. Hence, the client saves round trip time when NetScaler initiates the same request to the next available service. +For more information, see the AppQoE support documentation. [Appqoe resource sharing CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/appqoe.md) documentation. + +Example files: [appqoe-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/appqoe_example.yaml) + +#### WILDCARDDNS CRD: + +[WILDCARDDNS CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/wildcarddns.md) Wildcard DNS domains are used to handle requests for nonexistent domains and subdomains. In a zone, use wildcard domains to redirect queries for all nonexistent domains or subdomains to a particular server, instead of creating a separate Resource Record (RR) for each domain. The most common use of a wildcard DNS domain is to create a zone that can be used to forward mail from the internet to some other mail system. +For more information, see the Wild card DNS domains support documentation. [Wildcard DNS Entry CRD](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/crds/wildcarddns.md) documentation. + +Example files: [wildcarddns-crd.yaml](https://github.com/netscaler/netscaler-helm-charts/tree/master/example-crds/wildcarddns-example.yaml) + +### Tolerations + +Taints are applied on cluster nodes whereas tolerations are applied on pods. Tolerations enable pods to be scheduled on node with matching taints. For more information see [Taints and Tolerations in Kubernetes](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/). + +Toleration can be applied to NetScaler ingress controller pod using `tolerations` argument while deploying NSIC using helm chart. This argument takes list of tolerations that user need to apply on the NSIC pods. + +For example, following command can be used to apply toleration on the NSIC pod: + +``` +helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,tolerations[0].key=,tolerations[0].value=,tolerations[0].operator=,tolerations[0].effect= +``` + +Here tolerations[0].key, tolerations[0].value and tolerations[0].effect are the key, value and effect that was used while tainting the node. +Effect represents what should happen to the pod if the pod don't have any matching toleration. It can have values `NoSchedule`, `NoExecute` and `PreferNoSchedule`. +Operator represents the operation to be used for key and value comparison between taint and tolerations. It can have values `Exists` and `Equal`. The default value for operator is `Equal`. + +### Resource Quotas +There are various use-cases when resource quotas are configured on the Kubernetes cluster. If quota is enabled in a namespace for compute resources like cpu and memory, users must specify requests or limits for those values; otherwise, the quota system may reject pod creation. The resource quotas for the NSIC containers can be provided explicitly in the helm chart. + +To set requests and limits for the NSIC container, use the variables `resources.requests` and `resources.limits` respectively. + +Below is an example of the helm command that configures +- For NSIC container: +``` + CPU request for 500milli CPUs + CPU limit at 1000m + Memory request for 512M + Memory limit at 1000M +``` +``` +helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,nsVIP=,license.accept=yes,adcCredentialSecret=,resources.requests.cpu=500m,resources.requests.memory=512Mi --set resources.limits.cpu=1000m,resources.limits.memory=1000Mi +``` + +#### Analytics Configuration required for export of metrics to Prometheus + +If NetScaler VPX needs to send data to Prometheus directly without an exporter resource in between, then the below steps can be followed to install NetScaler ingress controller for NetScaler VPX. NSIC configures NetScaler VPX with the configuration required. + +1. Deploy NetScaler ingress controller using helm command: + +``` +helm repo add netscaler https://netscaler.github.io/netscaler-helm-charts/ + +helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,analyticsConfig.required=true,analyticsConfig.timeseries.metrics.enable=true,analyticsConfig.timeseries.port=5563,analyticsConfig.timeseries.metrics.mode=prometheus,analyticsConfig.timeseries.metrics.enableNativeScrape=true +``` + +2. For the NetScaler VPX to scrape using Prometheus, we need to create a system user with read only access. For more details on the user creation, refer [here](https://docs.netscaler.com/en-us/citrix-adc/current-release/observability/prometheus-integration#configure-read-only-prometheus-access-for-a-non-super-user) + +3. To setup Prometheus in order to scrape natively from NetScaler VPX, a new scrape job is required to be added under scrape_configs in the prometheus [configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/). A sample of the Prometheus job is given [here](https://docs.netscaler.com/en-us/citrix-adc/current-release/observability/prometheus-integration#prometheus-configuration) + +> **Note:** +> +> For more details on Prometheus integration, please refer to [this](https://docs.netscaler.com/en-us/citrix-adc/current-release/observability/prometheus-integration) + +### Configuration + +The following table lists the mandatory and optional parameters that you can configure during installation: + +| Parameters | Mandatory or Optional | Default value | Description | +| --------- | --------------------- | ------------- | ----------- | +| license.accept | Mandatory | no | Set `yes` to accept the NSIC end user license agreement. | +| imageRegistry | Mandatory | `quay.io` | The NetScaler ingress controller image registry | +| imageRepository | Mandatory | `netscaler/netscaler-k8s-ingress-controller` | The NetScaler ingress controller image repository | +| imageTag | Mandatory | `2.2.10` | The NetScaler ingress controller image tag | +| pullPolicy | Mandatory | IfNotPresent | The NSIC image pull policy. | +| imagePullSecrets | Optional | N/A | Provide list of Kubernetes secrets to be used for pulling the images from a private Docker registry or repository. For more information on how to create this secret please see [Pull an Image from a Private Registry](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). | +| nameOverride | Optional | N/A | String to partially override deployment fullname template with a string (will prepend the release name) | +| fullNameOverride | Optional | N/A | String to fully override deployment fullname template with a string | +| resources | Optional | {} | CPU/Memory resource requests/limits for NetScaler Ingress Controller container | +| adcCredentialSecret | Mandatory | N/A | The secret key to log on to the NetScaler VPX or MPX. For information on how to create the secret keys, see [Prerequisites](#prerequistes). | +| secretStore.enabled | Optional | False | Set to "True" for deploying other Secret Provider classes | +| secretStore.username | Optional | N/A | if `secretStore.enabled`, `username` of NetScaler will be fetched from the Secret Provider | +| secretStore.password | Optional | N/A | if `secretStore.enabled`, `password` of NetScaler will be fetched from the Secret Provider | +| nsIP | Mandatory | N/A | The IP address of the NetScaler device. For details, see [Prerequisites](#prerequistes). | +| nsVIP | Optional | N/A | The Virtual IP address on the NetScaler device. | +| nsSNIPS | Optional | N/A | The list of subnet IPAddresses on the NetScaler device, which will be used to create PBR Routes instead of Static Routes [PBR support](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/docs/how-to/pbr.md) | +| nsPort | Optional | 443 | The port used by NSIC to communicate with NetScaler. You can use port 80 for HTTP. | +| nsProtocol | Optional | HTTPS | The protocol used by NSIC to communicate with NetScaler. You can also use HTTP on port 80. | +| nsEnableLabel | Optional | True | Set to true for plotting Servicegraph. Ensure `analyticsConfig` are set. | +| nitroReadTimeout | Optional | 20 | The nitro Read timeout in seconds, defaults to 20 | +| logLevel | Optional | INFO | The loglevel to control the logs generated by NSIC. The supported loglevels are: CRITICAL, ERROR, WARNING, INFO, DEBUG, TRACE and NONE. For more information, see [Logging](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/configure/log-levels.md).| +| jsonLog | Optional | false | Set this argument to true if log messages are required in JSON format | +| nsConfigDnsRec | Optional | false | To enable/disable DNS address Record addition in NetScaler through Ingress | +| nsSvcLbDnsRec | Optional | false | To enable/disable DNS address Record addition in NetScaler through Type Load Balancer Service | +| nsDnsNameserver | Optional | N/A | To add DNS Nameservers in NetScaler | +| optimizeEndpointBinding | Optional | false | To enable/disable binding of backend endpoints to servicegroup in a single API-call. Recommended when endpoints(pods) per application are large in number. Applicable only for NetScaler Version >=13.0-45.7 | +| kubernetesURL | Optional | N/A | The kube-apiserver url that NSIC uses to register the events. If the value is not specified, NSIC uses the [internal kube-apiserver IP address](https://kubernetes.io/docs/tasks/access-application-cluster/access-cluster/#accessing-the-api-from-a-pod). | +| clusterName | Optional | N/A | The unique identifier of the kubernetes cluster on which the NSIC is deployed. Used in GSLB deployments. | +| ingressClass | Optional | N/A | If multiple ingress load balancers are used to load balance different ingress resources. You can use this parameter to specify NSIC to configure NetScaler associated with specific ingress class. For more information on Ingress class, see [Ingress class support](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/ingress-classes/). For Kubernetes version >= 1.19, this will create an IngressClass object with the name specified here | +| setAsDefaultIngressClass | Optional | False | Set the IngressClass object as default ingress class. New Ingresses without an "ingressClassName" field specified will be assigned the class specified in ingressClass. Applicable only for kubernetes versions >= 1.19 | +| serviceClass | Optional | N/A | By Default ingress controller configures all TypeLB Service on the NetScaler. You can use this parameter to finetune this behavior by specifing NSIC to only configure TypeLB Service with specific service class. For more information on Service class, see [Service class support]( https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/service-classes/). | +| nodeWatch | Optional | false | Use the argument if you want to automatically configure network route from the Ingress NetScaler VPX or MPX to the pods in the Kubernetes cluster. For more information, see [Automatically configure route on the NetScaler instance](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/network/staticrouting/#automatically-configure-route-on-the-citrix-adc-instance). | +| nsncPbr | Optional | False | Use this argument to inform NSIC that NetScaler Node Controller(NSNC) is configuring Policy Based Routes(PBR) on the NetScaler. For more information, see [NSNC-PBR-SUPPORT](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/pbr.md#configure-pbr-using-the-citrix-node-controller) | +| defaultSSLCertSecret | Optional | N/A | Provide Kubernetes secret name that needs to be used as a default non-SNI certificate in NetScaler. | +| defaultSSLSNICertSecret | Optional | N/A | Provide Kubernetes secret name that needs to be used as a default SNI certificate in NetScaler. | +| podIPsforServiceGroupMembers | Optional | False | By default NetScaler Ingress Controller will add NodeIP and NodePort as service group members while configuring type LoadBalancer Services and NodePort services. This variable if set to `True` will change the behaviour to add pod IP and Pod port instead of nodeIP and nodePort. Users can set this to `True` if there is a route between NetScaler and K8s clusters internal pods either using feature-node-watch argument or using NetScaler Node Controller. | +| ignoreNodeExternalIP | Optional | False | While adding NodeIP, as Service group members for type LoadBalancer services or NodePort services, NetScaler Ingress Controller has a selection criteria whereas it choose Node ExternalIP if available and Node InternalIP, if Node ExternalIP is not present. But some users may want to use Node InternalIP over Node ExternalIP even if Node ExternalIP is present. If this variable is set to `True`, then it prioritises the Node Internal IP to be used for service group members even if node ExternalIP is present | +| nsHTTP2ServerSide | Optional | OFF | Set this argument to `ON` for enabling HTTP2 for NetScaler service group configurations. | +| nsCookieVersion | Optional | 0 | Specify the persistence cookie version (0 or 1). | +| profileSslFrontend | Optional | N/A | Specify the frontend SSL profile. For Details see [Configuration using FRONTEND_SSL_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| profileTcpFrontend | Optional | N/A | Specify the frontend TCP profile. For Details see [Configuration using FRONTEND_TCP_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| profileHttpFrontend | Optional | N/A | Specify the frontend HTTP profile. For Details see [Configuration using FRONTEND_HTTP_PROFILE](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/configure/profiles.html#global-front-end-profile-configuration-using-configmap-variables) | +| ipam | Optional | False | Set this argument if you want to use the IPAM controller to automatically allocate an IP address to the service of type LoadBalancer. | +| disableAPIServerCertVerify | Optional | False | Set this parameter to True for disabling API Server certificate verification. | +| logProxy | Optional | N/A | Provide Elasticsearch or Kafka or Zipkin endpoint for NetScaler observability exporter. | +| entityPrefix | Optional | k8s | The prefix for the resources on the NetScaler VPX/MPX. | +| multiClusterPrefix | Optional | mc | The prefix for the shared resources on the NetScaler VPX/MPX for multicluster ingress feature. Ingress Controllers that are collaboratively sharing the csvserver IP should be configured with same value for multiclusterPrefix. For more information see [this](https://docs.netscaler.com/en-us/netscaler-k8s-ingress-controller/deploy/multicluster-ingress) | +| updateIngressStatus | Optional | True | Set this argurment if `Status.LoadBalancer.Ingress` field of the Ingress resources managed by the NetScaler ingress controller needs to be updated with allocated IP addresses. For more information see [this](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/configure/ingress-classes.md#updating-the-ingress-status-for-the-ingress-resources-with-the-specified-ip-address). | +| routeLabels | Optional | proxy in () | You can use this parameter to provide the route labels selectors to be used by NetScaler Ingress Controller for routeSharding in OpenShift cluster. | +| namespaceLabels | Optional | N/A | You can use this parameter to provide the namespace labels selectors to be used by NetScaler Ingress Controller for routeSharding in OpenShift cluster. | +| podAnnotations | Optional | N/A | Map of annotations to add to the pods. | +| affinity | Optional | N/A | Affinity labels for pod assignment. | +| exporter.required | Optional | false | Use the argument, if you want to run the [Exporter for NetScaler Stats](https://github.com/netscaler/netscaler-adc-metrics-exporter) along with NSIC to pull metrics for the NetScaler VPX or MPX| +| exporter.imageRegistry | Optional | `quay.io` | The Exporter for NetScaler Stats image registry | +| exporter.imageRepository | Optional | `netscaler/netscaler-adc-metrics-exporter` | The Exporter for NetScaler Stats image repository | +| exporter.imageTag | Optional | `1.4.9` | The Exporter for NetScaler Stats image tag | +| exporter.pullPolicy | Optional | IfNotPresent | The Exporter image pull policy. | +| exporter.ports.containerPort | Optional | 8888 | The Exporter container port. | +| exporter.resources | Optional | {} | CPU/Memory resource requests/limits for Metrics exporter container | +| exporter.extraVolumeMounts | Optional | [] | Specify the Additional VolumeMounts to be mounted in Exporter container. Specify the volumes in `extraVolumes` | +| exporter.serviceMonitorExtraLabels | Optional | | Extra labels for service monitor whem NetScaler-adc-metrics-exporter is enabled. | +| openshift | Optional | false | Set this argument if OpenShift environment is being used. | +| disableOpenshiftRoutes | Optional | false | By default Openshift routes are processed in openshift environment, this variable can be used to disable Ingress controller processing the openshift routes. | +| nodeSelector.key | Optional | N/A | Node label key to be used for nodeSelector option in NSIC deployment. | +| nodeSelector.value | Optional | N/A | Node label value to be used for nodeSelector option in NSIC deployment. | +| tolerations | Optional | N/A | Specify the tolerations for the NSIC deployment. | +| analyticsConfig.required | Mandatory | false | Set this to true if you want to configure NetScaler to send metrics and transaction records to analytics . | +| analyticsConfig.distributedTracing.enable | Optional | false | Set this value to true to enable OpenTracing in NetScaler. | +| analyticsConfig.distributedTracing.samplingrate | Optional | 100 | Specifies the OpenTracing sampling rate in percentage. | +| analyticsConfig.endpoint.metrics.service | Optional | N/A | Set this value as the IP address or DNS address of the analytics server. Format: servicename.namespace, servicename.namespace.svc.cluster.local, namespace/servicename *** This value replaces the analyticsConfig.endpoint.server value used earlier. *** | +| analyticsConfig.endpoint.transactions.service | Optional | N/A | Set this value as the IP address or service name with namespace of the analytics service deployed in k8s environment. Format: namespace/servicename *** This value replaces the analyticsConfig.endpoint.service value used earlier. *** | +| analyticsConfig.timeseries.port | Optional | 30002 | Specify the port used to expose analytics service outside cluster for timeseries endpoint. | +| analyticsConfig.timeseries.metrics.enable | Optional | False | Set this value to true to enable sending metrics from NetScaler. | +| analyticsConfig.timeseries.metrics.mode | Optional | avro | Specifies the mode of metric endpoint. | +| analyticsConfig.timeseries.metrics.exportFrequency | Optional | 30 | Specifies the time interval for exporting time-series data. Possible values range from 30 to 300 seconds. | +| analyticsConfig.timeseries.metrics.schemaFile | Optional | schema.json | Specifies the name of a schema file with the required Netscaler counters to be added and configured for metricscollector to export. A reference schema file reference_schema.json with all the supported counters is also available under the path /var/metrics_conf/. This schema file can be used as a reference to build a custom list of counters. | +| analyticsConfig.timeseries.metrics.enableNativeScrape | Optional | false | Set this value to true for native export of metrics. | +| analyticsConfig.timeseries.auditlogs.enable | Optional | false | Set this value to true to export audit log data from NetScaler. | +| analyticsConfig.timeseries.events.enable | Optional | false | Set this value to true to export events from the NetScaler. | +| analyticsConfig.transactions.enable | Optional | false | Set this value to true to export transactions from NetScaler. | +| analyticsConfig.transactions.port | Optional | 30001 | Specify the port used to expose analytics service outside cluster for transaction endpoint. | +| nsLbHashAlgo.required | Optional | false | Set this value to set the LB consistent hashing Algorithm | +| nsLbHashAlgo.hashFingers | Optional | 256 | Specifies the number of fingers to be used for hashing algorithm. Possible values are from 1 to 1024, Default value is 256 | +| nsLbHashAlgo.hashAlgorithm | Optional | 'default' | Specifies the supported algorithm. Supported algorithms are "default", "jarh", "prac", Default value is 'default' | +| extraVolumeMounts | Optional | [] | Specify the Additional VolumeMounts to be mounted in NSIC container | +| extraVolumes | Optional | [] | Specify the Additional Volumes for additional volumeMounts | +| rbacRole | Optional | false | To deploy NSIC with RBAC Role set rbacRole=true; by default NSIC gets installed with RBAC ClusterRole(rbacRole=false)) | +| nodeLabels | Optional | "" | If there are pods on nodes in the nodes with this nodeLabels, NSIC will configure NetScaler to advertises the VIP using BGP | +| bgpAdvertisement | Optional | False | To advertise VIP using BGP from NetScaler | +| enableLivenessProbe | Optional | True | Enable LivenessProbes settings for NetScaler Ingress Controller | +| enableReadinessProbe | Optional | True | Enable LivenessProbes settings for NetScaler Ingress Controller | +| readinessProbe | Optional | N/A | Set readinessProbe settings NetScaler Ingress Controller | +| livenessProbe| Optional | N/A | Set livenessPorbe settings for NetScaler Ingress Controller | + + +Alternatively, you can define a YAML file with the values for the parameters and pass the values while installing the chart. + +For example: + ``` + helm install my-release netscaler/netscaler-ingress-controller -f values.yaml + ``` + +> **Tip:** +> +> The [values.yaml](https://github.com/netscaler/netscaler-helm-charts/blob/master/netscaler-ingress-controller/values.yaml) contains the default values of the parameters. + +> **Note:** +> +> Please provide frontend-ip (VIP) in your application ingress yaml file. For more info refer [this](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/configure/annotations.md). + +## Route Addition in MPX/VPX +For seamless functioning of services deployed in the Kubernetes cluster, it is essential that Ingress NetScaler device should be able to reach the underlying overlay network over which Pods are running. +`feature-node-watch` knob of NetScaler Ingress Controller can be used for automatic route configuration on NetScaler towards the pod network. Refer [Static Route Configuration](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/staticrouting.md) for further details regarding the same. +By default, `feature-node-watch` is false. It needs to be explicitly set to true if auto route configuration is required. + +This can also be achieved by deploying [NetScaler Node Controller](https://github.com/netscaler/netscaler-k8s-node-controller). + +If your deployment uses one single NetScaler Device to loadbalance between multiple k8s clusters, there is a possibilty of CNI subnets to overlap, causing the above mentioned static routing to fail due to route conflicts. In such deployments [Policy Based Routing(PBR)] ( https://docs.netscaler.com/en-us/citrix-adc/current-release/networking/ip-routing/configuring-policy-based-routes/configuring-policy-based-routes-pbrs-for-ipv4-traffic.html) can be used instead. This would require you to provide one or more subnet IP Addresses unique for each kubernetes cluster either via Environment variable or Configmap, see [PBR Support](https://github.com/netscaler/netscaler-k8s-ingress-controller/tree/master/docs/how-to/pbr.md) + + Use the following command to provide subnet IPAddresses(SNIPs) to configure Policy Based Routes(PBR) on the NetScaler + + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,nsSNIPS='[\, \, ...]' + ``` + + [NetScaler Node Controller](https://github.com/netscaler/netscaler-k8s-node-controller) by default also adds static routes while creating the VXLAN tunnel. To use [Policy Based Routing(PBR)] ( https://docs.netscaler.com/en-us/citrix-adc/current-release/networking/ip-routing/configuring-policy-based-routes/configuring-policy-based-routes-pbrs-for-ipv4-traffic.html) to avoid static route clash, both NetScaler Node Controller and NetScaler Ingress Controller has to work in conjunction and has to be started with specific arguments. For more details refer [NSNC-PBR-SUPPORT](https://github.com/netscaler/netscaler-k8s-ingress-controller/blob/master/docs/network/pbr.md#configure-pbr-using-the-citrix-node-controller). + + Use the following command to inform NetScaler Ingress Controller that NetScaler Node Controller is configuring Policy Based Routes(PBR) on the NetScaler + + ``` + helm install my-release netscaler/netscaler-ingress-controller --set nsIP=,license.accept=yes,adcCredentialSecret=,clusterName=,nsncPbr= + ``` + +For configuring static routes manually on NetScaler VPX or MPX to reach the pods inside the cluster follow: +### For Kubernetes: +1. Obtain podCIDR using below options: + ``` + kubectl get nodes -o yaml | grep podCIDR + ``` + * podCIDR: 10.244.0.0/24 + * podCIDR: 10.244.1.0/24 + * podCIDR: 10.244.2.0/24 + +2. Log on to the NetScaler instance. + +3. Add Route in Netscaler VPX/MPX + ``` + add route + ``` +4. Ensure that Ingress MPX/VPX has a SNIP present in the host-network (i.e. network over which K8S nodes communicate with each other. Usually eth0 IP is from this network). + + Example: + * Node1 IP = 192.0.2.1 + * podCIDR = 10.244.1.0/24 + * add route 10.244.1.0 255.255.255.0 192.0.2.1 + +### For OpenShift: +1. Use the following command to get the information about host names, host IP addresses, and subnets for static route configuration. + ``` + oc get hostsubnet + ``` + +2. Log on to the NetScaler instance. + +3. Add the route on the NetScaler instance using the following command. + ```add route ``` + +4. Ensure that Ingress MPX/VPX has a SNIP present in the host-network (i.e. network over which OpenShift nodes communicate with each other. Usually eth0 IP is from this network). + + For example, if the output of the `oc get hostsubnet` is as follows: + * oc get hostsubnet + + NAME HOST HOST IP SUBNET + os.example.com os.example.com 192.0.2.1 10.1.1.0/24 + + * The required static route is as follows: + + add route 10.1.1.0 255.255.255.0 192.0.2.1 + +## Uninstalling the Chart +To uninstall/delete the ```my-release``` deployment: + + ``` + helm delete my-release + ``` + +The command removes all the Kubernetes components associated with the chart and deletes the release. + +## Related documentation + +- [NetScaler ingress controller Documentation](https://docs.netscaler.com/en-us/citrix-k8s-ingress-controller/) +- [NetScaler ingress controller GitHub](https://github.com/netscaler/netscaler-k8s-ingress-controller) diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/app-readme.md b/charts/netscaler/netscaler-ingress-controller/2.2.10/app-readme.md new file mode 100644 index 000000000..07cb91739 --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/app-readme.md @@ -0,0 +1,5 @@ +# NetScaler Ingress Controller + +[NetScaler Ingress Controller](https://github.com/netscaler/netscaler-k8s-ingress-controller) is an ingress controller for NetScaler MPX (hardware), NetScaler VPX (virtualized), and NetScaler CPX (containerized) for bare metal and cloud deployments. It is built around Kubernetes Ingress and automatically configures NetScaler based on the Ingress resource configuration. + +This chart bootstraps standalone NetScaler Ingress Controller which can be used to configure NetScaler MPX or VPX. diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/crds/crds.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/crds/crds.yaml new file mode 100644 index 000000000..9db6a9972 --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/crds/crds.yaml @@ -0,0 +1,2706 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: rewritepolicies.citrix.com +spec: + group: citrix.com + names: + kind: rewritepolicy + plural: rewritepolicies + singular: rewritepolicy + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + rewrite-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to rewrite policy.' + type: array + items: + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be + evaluated if the current policy evaluates to TRUE. + Specify one of the following values: + * NEXT - Evaluate the policy with the next higher priority number. + * END - End policy evaluation. + Default value of goto-priority-expression: END' + type: string + maxLength: 1499 + logpackets: + type: object + description: 'Adds an audit message action. + The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + rewrite-policy: + type: object + properties: + rewrite-criteria: + description: 'Expression against which traffic is evaluated.' + type: string + maxLength: 1299 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). + An UNDEF event indicates an internal error condition.' + type: string + maxLength: 77 + enum: ['NOREWRITE', 'RESET', 'DROP'] + operation: + description: 'Type of user-defined rewrite action.' + type: string + enum: ["noop", "delete", "insert_http_header", "delete_http_header", + "corrupt_http_header", "insert_before", "insert_after", "replace", + "replace_http_res", "delete_all", "replace_all", "insert_before_all", + "insert_after_all", "clientless_vpn_encode", "clientless_vpn_encode_all", + "clientless_vpn_decode", "clientless_vpn_decode_all", "insert_sip_header", + "delete_sip_header", "corrupt_sip_header", "replace_sip_res", "replace_diameter_header_field", + "replace_dns_header_field", "replace_dns_answer_section"] + target: + description: 'Default syntax expression that specifies which part of the request or response to rewrite.' + type: string + maxLength: 1229 + modify-expression: + description: 'Default syntax expression that specifies the content to insert into the request + or response at the specified location, or that replaces the specified string.' + type: string + maxLength: 7991 + multiple-occurence-modify: + description: 'Search facility that is used to match multiple strings in the request or response.' + type: string + maxLength: 171 + additional-multiple-occurence-modify: + description: 'Specify additional criteria to refine the results of the search. + Always starts with the "extend(m,n)" operation, where "m" specifies number of bytes to the left of selected data + and "n" specifies number of bytes to the right of selected data. + You can use refineSearch only on body expressions, and only when rewrite-criteria is any one of this: + INSERT_BEFORE_ALL, INSERT_AFTER_ALL, REPLACE_ALL, and DELETE_ALL.' + type: string + maxLength: 1299 + direction: + description: 'Bind point to which to bind the policy.' + type: string + enum: ["REQUEST","RESPONSE"] + comment: + description: 'Any comments to preserve information about this rewrite policy.' + type: string + maxLength: 255 + required: [rewrite-criteria, operation, target, direction] + required: [rewrite-policy] + + responder-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to responder policy.' + type: array + items: + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be + evaluated if the current policy evaluates to TRUE. + Specify one of the following values: + * NEXT - Evaluate the policy with the next higher priority number. + * END - End policy evaluation. + Default value of goto-priority-expression: END' + type: string + maxLength: 1499 + logpackets: + type: object + description: 'Adds an audit message action. + The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", + "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + responder-policy: + type: object + properties: + redirect: + type: object + description: 'Use this option when you want to Redirect the request when request matches to policy.' + properties: + url: + description: 'URL on which you want to redirect the request.' + type: string + maxLength: 7991 + redirect-status-code: + description: 'HTTP response status code, for example 200, 302, 404, etc.' + type: integer + minimum: 100 + maximum: 599 + redirect-reason: + description: 'Expression specifying the reason for redirecting the request.' + type: string + maxLength: 7991 + required: [url] + respondwith: + type: object + description: 'Use this parameter when you want to respond to the request when request matches to policy.' + properties: + http-payload-string: + description: 'Expression that you want to sent as response to the request.' + type: string + maxLength: 7991 + required: [http-payload-string] + noop: + type: string + description: 'Use this option when you want to send the request to the protected server instead of + responding to it when request matches to policy.' + properties: + target: + description: 'Default syntax expression that specifies to perform noop operation on' + type: string + maxLength: 1229 + reset: + type: string + description: 'Use this option when you want to Reset the client connection by closing it when request matches to policy.' + properties: + drop: + type: string + description: 'Use this option when you want to drop the request without sending a response to the user when request matches to policy.' + properties: + respond-criteria: + description: 'Default syntax expression that the policy uses to determine whether to respond to the specified request.' + type: string + maxLength: 1299 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). + An UNDEF event indicates an internal error condition.' + type: string + maxLength: 77 + enum: ['NOOP', 'RESET', 'DROP'] + comment: + description: 'Any comments to preserve information about this responder policy.' + type: string + maxLength: 255 + required: [respond-criteria] + oneOf: [required: [redirect], required: [respondwith], required: [noop], required: [reset], required: [drop]] + required: [responder-policy] + + dataset: + type: array + items: + type: object + properties: + name: + description: 'Name of the dataset.' + type: string + maxLength: 32 + type: + description: 'Type of value to bind to the dataset.' + type: string + enum: ["ipv4", "number", "ipv6", "ulong", "double", "mac"] + comment: + description: 'Any comments to preserve information about this dataset.' + type: string + maxLength: 255 + values: + description: 'Value of the specified type that is associated with this dataset.' + type: array + items: + type: string + required: [name, type, values] + + patset: + type: array + items: + type: object + properties: + name: + description: 'Name of the Patset.' + type: string + maxLength: 32 + comment: + description: 'Any comments to preserve information about this patset.' + type: string + maxLength: 255 + values: + description: 'String of characters that constitutes a pattern and is associated with this patset.' + type: array + items: + type: string + required: [name, values] + + stringmap: + type: array + items: + type: object + properties: + name: + description: 'Name of the Stringmap.' + type: string + maxLength: 32 + comment: + description: 'Any comments to preserve information about this stringmap.' + type: string + maxLength: 255 + values: + description: 'List of (key,value) pairs to be bound to this string map.' + type: array + items: + type: object + properties: + key: + description: 'Character string constituting the key to be bound to this string map.' + type: string + maxLength: 2047 + value: + description: 'Character string constituting the value associated with the key.' + type: string + maxLength: 2047 + required: [name, values] + + httpcallout_policy: + type: array + items: + type: object + properties: + name: + description: 'httpcallout name' + type: string + maxLength: 32 + server_ip: + description: 'IP Address of the server(callout agent) to which the callout is sent.' + type: string + server_port: + description: 'Port of the server(callout agent) to which the callout is sent.' + type: integer + minimum: 1 + maximum: 65535 + http_method: + description: |+ + 'Method used in the HTTP request that this callout sends. + Default http method is GET' + type: string + enum: ['GET', 'POST'] + host_expr: + description: |+ + 'String expression to configure the Host header. Can contain a literal value + (for example, 10.101.10.11) or a derived value (for example, http.req.header("Host")). + The literal value can be an IP address or a fully qualified domain name. Mutually + exclusive with the full HTTP request expression.' + type: string + maxLength: 255 + url_stem_expr: + description: |+ + 'String expression for generating the URL stem. Can contain a literal string + (for example, "/mysite/index.html") or an expression that derives the value + (for example, http.req.url).' + type: string + maxLength: 8191 + headers: + type: array + description: |+ + 'One or more headers to insert into the HTTP request. Each header is represented by + name and expr, where expr is an expression that is evaluated at runtime to provide + the value for the named header. You can configure a maximum of eight headers for + an HTTP callout.' + items: + type: object + properties: + name: + description: 'header name' + type: string + expr: + description: 'header expression' + type: string + parameters: + type: array + description: |+ + 'One or more query parameters to insert into the HTTP request URL (for a GET request) + or into the request body (for a POST request). Each parameter is represented by + name and expr, where expr is an expression that is evaluated at run time to provide + the value for the named parameter (name=value). The parameter values are URL encoded.' + items: + type: object + properties: + name: + description: 'parameter name' + type: string + expr: + description: 'parameter expression' + type: string + body_expr: + description: |+ + 'An advanced string expression for generating the body of the request. + The expression can contain a literal string or an expression that derives + the value (for example, client.ip.src).' + type: string + full_req_expr: + description: |+ + 'Exact HTTP request, in the form of an expression, which the NetScaler sends + to the callout agent. The request expression is constrained by the feature + for which the callout is used. For example, an HTTP.RES expression cannot be + used in a request-time policy bank or in a TCP content switching policy bank.' + type: string + scheme: + description: |+ + 'Type of scheme for the callout server. + Default scheme is HTTP' + type: string + enum: ['HTTP', 'HTTPS'] + cache_for_secs: + description: |+ + 'Duration, in seconds, for which the callout response is cached. + The cached responses are stored in an integrated caching content + group named "calloutContentGroup". If no duration is configured, + the callout responses will not be cached unless normal caching + configuration is used to cache them. This parameter takes precedence over any + normal caching configuration that would otherwise apply to these responses.' + type: integer + minimum: 1 + maximum: 31536000 + return_type: + description: |+ + 'Type of data that the target callout agent returns in response to the callout + Available settings function as follows: + * TEXT - Treat the returned value as a text string. + * NUM - Treat the returned value as a number. + * BOOL - Treat the returned value as a Boolean value.' + type: string + enum: ['TEXT', 'NUM', 'BOOL'] + result_expr: + description: |+ + 'Expression that extracts the callout results from the response sent by the HTTP callout + agent. Must be a response based expression, that is, it must begin with HTTP.RES. The + operations in this expression must match the return type. For example, if you configure + a return type of TEXT, the result expression must be a text based expression. If the + return type is NUM, the result expression (resultExpr) must return a numeric value, + as in the following example: http.res.body(10000).length.' + type: string + maxLength: 8191 + comment: + description: 'Any comments to preserve information about this HTTP callout.' + type: string + maxLength: 255 + allOf: + - properties: + required: [name, server_ip, server_port] + - properties: + oneOf: + - properties: + required: [full_req_expr] + - properties: + anyOf: + - properties: + required: [http_method] + - properties: + required: [host_expr] + - properties: + required: [url_stem_expr] + - properties: + required: [headers] + - properties: + required: [parameters] + - properties: + required: [body_expr] + anyOf: [required: [rewrite-policies], required: [responder-policies]] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: ratelimits.citrix.com +spec: + group: citrix.com + names: + kind: ratelimit + plural: ratelimits + singular: ratelimit + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the ratelimit policies are applied.' + type: array + items: + type: string + maxLength: 127 + selector_keys: + type: object + description: 'Traffic match criteria to which apply above rate-limit/throttling. All keys are applied as AND condition. If no keys are specified, rate-limit applies at service level' + properties: + basic: + type: object + description: "Basic traffic stream selection criteria to which to apply the ratelimit" + properties: + path: + type: array + description: "api resource path prefix match. e.g. /api/v1/products" + items: + type: string + method: + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header_name: + description: "HTTP header that identifies the unique API client for e.g. X-apikey" + type: string + per_client_ip: + description: "Setting this applies the throttling limit to each unique Client IP address accessing the API resource" + type: boolean + req_threshold: + description: 'Max requests per timeslice units to be allowed' + type: integer + timeslice: + description: 'Timeslice in miliseconds in multiple of 10. Defaults to 1000 miliseconds' + type: integer + limittype: + description: "Burst mode or smooth. Defaults to smooth limittype if not specified" + type: string + enum: ['BURSTY','SMOOTH'] + throttle_action: + type: string + enum: ['DROP', 'RESET','REDIRECT', 'RESPOND'] + description: "Drop will drop the requests exceeding limits, RESET will reset the client connection, Redirect will redirect to specified URL, respond will respond with 429 'Exceeded allowed rate of requests'" + redirect_url: + type: string + description: "Redirect-URL" + logpackets: + type: object + description: 'Adds an audit message action. The action specifies whether to log the message, and to which log.' + properties: + logexpression: + description: 'Default-syntax expression that defines the format and content of the log message.' + type: string + maxLength: 7991 + loglevel: + description: 'Audit log level, which specifies the severity level of the log message being generated.' + type: string + enum: ["EMERGENCY", "ALERT", "CRITICAL", "ERROR", "WARNING", "NOTICE", "INFORMATIONAL", "DEBUG"] + required: [logexpression, loglevel] + required: [req_threshold] +--- +#Sample CRD instance + +#apiVersion: citrix.com/v1 +#description: VIP for apache service +#kind: vip +#metadata: +# name: service-apache +# namespace: default +#spec: +# description: VIP for the apache Service +# ipaddress: 10.99.98.90 +# kind: service +# name: apache + +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: vips.citrix.com +spec: + group: citrix.com + names: + kind: vip + plural: vips + singular: vip + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + additionalPrinterColumns: + - jsonPath: .spec.ipaddress + name: VIP + type: string + - name: Age + type: date + jsonPath: .metadata.creationTimestamp + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + properties: + ipaddress: + type: string + name: + type: string + kind: + type: string + enum: ["service", "ingress", "listener"] + description: + type: string + range-name: + type: string + multicluster: + description: "The setting of this indicates that the VIP/csvserver IP address is shared by multiple netscaler ingress controllers on the VPX/MPX. For CPX, this field is not applicable" + type: boolean +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: authpolicies.citrix.com +spec: + group: citrix.com + names: + kind: authpolicy + plural: authpolicies + singular: authpolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: 'Current Status of the CRD' + jsonPath: .status.state + - name: Message + type: string + description: 'Status Message' + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: |+ + 'Name of the services for which the policies applied' + type: array + items: + type: string + maxLength: 63 + authentication_mechanism: + type: object + description: |+ + 'Authentication mechanism. Options: using forms or using request header. + Default is Authentication using request header, when no option is specified' + properties: + using_request_header: + description: |+ + 'Enable user authentication using request header. Use when the credentials + or api keys are passed in a header. For example, when using Basic, Digest, + Bearer authentication or api keys. + When authentication using forms is provided, this is set to OFF' + + type: string + using_forms: + type: object + description: 'Enables authentication using forms. Use with user/web authentication.' + properties: + authentication_host: + description: |+ + 'Fully qualified domain name (FQDN) for authentication. + This FQDN should be unique and should resolve to frontend IP of + NetScaler with Ingress/service type LoadBalancer (or) vip of Listener CRD' + type: string + maxLength: 255 + authentication_host_cert: + description: |+ + 'Name of the SSL certificate to be used with authentication_host. + This certificate is mandatory while using_forms' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - properties: + required: [tls_secret] + - properties: + required: [preconfigured] + ingress_name: + description: |+ + 'Ingress name for which the authentication using forms + is applicable.' + type: string + maxLength: 63 + lb_service_name: + description: |+ + 'Service of type LoadBalancer for which the authentication using forms + is applicable.' + type: string + maxLength: 63 + listener_name: + description: |+ + 'Listener CRD name for which the authentication using forms is applicable.' + type: string + maxLength: 63 + vip: + description: |+ + 'Frontend IP of ingress for which the authentication + using forms is applicable. This refers to frontend-ip provided + with Ingress. It is suggested to use vip, if more than one Ingress + resource use the same frontend-ip' + type: string + required: [authentication_host, authentication_host_cert] + oneOf: + - properties: + required: [ingress_name] + - properties: + required: [lb_service_name] + - properties: + required: [listener_name] + - properties: + required: [vip] + oneOf: + - properties: + using_request_header: + enum: ['ON'] + required: [using_request_header] + - properties: + required: [using_forms] + + authentication_providers: + description: |+ + 'Authentication Configuration for required authentication providers/schemes. + One or more of these can be created' + type: array + items: + description: 'Create config for a single authentication provider of a particular type' + type: object + properties: + name: + description: 'Name for this provider, has to be unique, referenced by authentication policies' + type: string + maxLength: 127 + + oauth: + description: 'Authentication provided by external oAuth provider' + type: object + properties: + issuer: + description: 'Identity of the server whose tokens are to be accepted' + type: string + maxLength: 127 + audience: + description: 'Audience for which token sent by Authorization server is applicable' + type: array + items: + type: string + maxLength: 127 + jwks_uri: + description: |+ + 'URL of the endpoint that contains JWKs (Json Web Key) for + JWT (Json Web Token) verification' + type: string + maxLength: 127 + introspect_url: + description: ' URL of the introspection server' + type: string + maxLength: 127 + client_credentials: + description: |+ + 'secrets object that contains Client Id and secret as known + to Introspection server' + type: string + maxLength: 253 + token_in_hdr: + description: |+ + 'custom header name where token is present, + default is Authorization header' + type: array + items: + type: string + maxLength: 127 + maxItems: 2 + token_in_param: + description: 'query parameter name where token is present' + type: array + items: + type: string + maxLength: 127 + maxItems: 2 + signature_algorithms: + description: 'list of allowed signature algorithms, by default HS256, RS256, RS512 are allowed' + type: array + items: + type: string + enum: ['HS256', 'RS256', 'RS512'] + claims_to_save: + description: 'list of claims to be saved, used to create authorization policies' + type: array + items: + type: string + maxLength: 127 + metadata_url: + description: 'URL used to get OAUTH/OIDC provider metadata' + type: string + maxLength: 255 + user_field: + description: |+ + 'Attribute in the token from which username should be extracted. + by default, NetScaler looks at email attribute for user id' + type: string + maxLength: 127 + default_group: + description: |+ + 'group assigned to the request if authentication succeeds, + this is in addition to any extracted groups from token' + type: string + maxLength: 63 + grant_type: + description: 'used to specify the type of flow to the token end point, defaults to CODE' + type: array + items: + type: string + enum: ['CODE','PASSWORD'] + pkce: + description: 'specify whether to enable Proof Key Code Exchange, defaults to ENABLED' + type: string + enum: ['ENABLED', 'DISABLED'] + token_ep_auth_method: + description: |+ + 'authentication method to be used with token end point, + defaults to client_secret_post' + type: string + enum: ['client_secret_post', 'client_secret_jwt'] + + anyOf: + - properties: + required : [jwks_uri] + - properties: + required : [introspect_url, client_credentials] + - properties: + required : [metadata_url] + + ldap: + description: 'LDAP authentication provider' + type: object + properties: + server_ip: + description: 'IP address assigned to the LDAP server' + type: string + server_name: + description: 'LDAP server name as a FQDN' + type: string + maxLength: 127 + server_port: + description: 'Port on which the LDAP server accepts connections. Default is 389' + type: integer + minimum: 1 + maximum: 65535 + base: + description: |+ + 'Base (node) from which to start LDAP searches. If the LDAP server is + running locally, the default value of base is dc=netscaler, dc=com' + type: string + maxLength: 127 + server_login_credentials: + description: |+ + 'Kubernetes secret object providing credentials to login to LDAP server, + The secret data should have username and password' + type: string + login_name: + description: |+ + 'LDAP login name attribute. The NetScaler uses the LDAP login name + to query external LDAP servers or Active Directories' + type: string + maxLength: 127 + security_type: + description: |+ + 'Type of security used for communications between the NetScaler + and the LDAP server. Default is TLS' + type: string + enum: ['PLAINTEXT', 'TLS', 'SSL'] + validate_server_cert: + description: 'Validate LDAP Server certs. Default is NO' + type: string + enum: ['YES', 'NO'] + hostname: + description: |+ + 'Hostname for the LDAP server. If validate_server_cert is ON, + this must be the host name on the certificate from the LDAP + A hostname mismatch will cause a connection failure' + type: string + maxLength: 127 + sub_attribute_name: + description: 'LDAP group sub-attribute name. Used for group extraction from the LDAP server.' + type: string + maxLength: 31 + group_attribute_name: + description: 'LDAP group attribute name. Used for group extraction on the LDAP server.' + type: string + maxLength: 31 + search_filter: + description: |+ + 'String to be combined with the default LDAP user search string to form the + search value. For example, if the search filter "vpnallowed=true" is combined + with the LDAP login name "samaccount" and the user-supplied username is "bob", + the result is the LDAP search string ""(&(vpnallowed=true)(samaccount=bob)"" + (Be sure to enclose the search string in two sets of double quotation marks)' + type: string + maxLength: 255 + auth_timeout: + description: |+ + 'Number of seconds the NetScaler waits for a response from the server + Default is 3' + type: integer + minimum: 1 + maximum: 4294967295 + password_change: + description: 'Allow password change requests. Default is DISABLED' + type: string + enum: ['ENABLED', 'DISABLED'] + attributes_to_save: + description: |+ + 'List of attribute names separated by comma which needs to be fetched + from LDAP server and stored as key-value pair for the session on NetScaler' + type: string + maxLength: 2047 + oneOf: + - properties: + required: [server_ip] + - properties: + required: [server_name] + + saml: + description: |+ + 'SAML authentication provider. + Currently SAML is supported only with authentication mechanism using forms' + type: object + properties: + metadata_url: + description: 'URL is used for obtaining saml metadata.' + type: string + maxLength: 255 + metadata_refresh_interval: + description: |+ + 'Interval in minutes for fetching metadata from specified metadata URL. + Default is 36000' + type: integer + minimum: 1 + maximum: 4294967295 + signing_cert: + description: 'SSL certificate to sign requests from SP to IDP' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - properties: + required: [tls_secret] + - properties: + required: [preconfigured] + audience: + description: 'Audience for which assertion sent by IdP is applicable' + type: string + maxLength: 127 + issuer_name: + description: 'The name to be used in requests sent from SP to IDP to identify NetScaler' + type: string + maxLength: 63 + binding: + description: 'Specifies the transport mechanism of saml message. Default is POST' + type: string + enum: ['REDIRECT', 'POST', 'ARTIFACT'] + artifact_resolution_service_url: + description: 'URL of the Artifact Resolution Service on IdP' + type: string + maxLength: 255 + logout_binding: + description: 'Specifies the transport mechanism of saml logout. Default is POST' + type: string + enum: ['REDIRECT', 'POST'] + reject_unsigned_assertion: + description: |+ + 'Reject unsigned SAML assertions. ON, rejects assertion without signature. + STRICT ensure that both Response and Assertion are signed. Default is ON' + type: string + enum: ['ON', 'OFF', 'STRICT'] + user_field: + description: 'SAML user ID, as given in the SAML assertion' + type: string + maxLength: 63 + default_authentication_group: + description: |+ + 'This is the default group that is chosen when the authentication + succeeds in addition to extracted groups' + type: string + maxLength: 63 + skew_time: + description: |+ + 'Allowed clock skew in number of minutes on an incoming assertion. + Default is 5' + type: integer + minimum: 1 + attributes_to_save: + description: |+ + 'List of attribute names separated by comma which needs to be extracted + and stored as key-value pair for the session on NetScaler' + type: string + maxLength: 2047 + required: + - metadata_url + + basic_local_db: + type: object + description: |+ + 'Basic HTTP authentication supported by NetScaler, user data in local DB of NetScaler. + Users needs to be added on NetScaler' + properties: + use_local_auth: + description: 'Use NetScaler authentication' + type: string + enum: ['YES'] + + required: + - name + + authentication_policies: + description: 'Authentication policies' + type: array + items: + type: object + description: 'Authentication policy' + properties: + resource: + type: object + description: 'endpoint/resource selection criteria' + properties: + path: + description: 'api resource path e.g. /products. ' + type: array + items: + type: string + maxLength: 511 + method: + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + required: + - path + expression: + description: 'NetScaler syntax expression for authentication' + type: string + maxLength: 1229 + provider: + description: 'name of the authentication provider for the policy, empty if no authentication required' + type: array + items: + type: string + maxLength: 127 + maxItems: 1 + oneOf: + - required: [resource, provider] + - required: [expression, provider] + + authorization_policies: + description: 'Authorization policies' + type: array + items: + type: object + description: 'Authorization policy' + properties: + resource: + type: object + description: 'endpoint/resource selection criteria' + properties: + path: + description: 'api resource path e.g. /products. ' + type: array + items: + type: string + maxLength: 511 + method: + description: ' http method' + type: array + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + claims: + description: 'authorization scopes required for selected resource saved as claims or attributes' + type: array + items: + type: object + properties: + name: + description: 'name of the claim/attribute to check' + type: string + maxLength: 127 + values: + description: 'list of claim values required for the request' + type: array + items: + type: string + maxLength: 127 + minItems: 1 + required: + - name + - values + required: + - claims + expression: + description: 'NetScaler syntax expression for authorization' + type: string + maxLength: 1229 + oneOf: + - required: [resource] + - required: [expression] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: listeners.citrix.com +spec: + group: citrix.com + names: + kind: Listener + plural: listeners + singular: listener + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + required: [spec] + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + required: [protocol] + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + multicluster: + description: "The setting of this indicates that the VIP/csvserver IP address is shared by multiple netscaler ingress controllers on the VPX/MPX. For CPX, this field is not applicable" + type: boolean + protocol: + type: string + enum: ["udp", "tcp", "https", "http"] + description: "Protocol for this listener" + vip: + type: string + description: "VIP address, Optional for CPX, required for Tier-1 deployments" + secondaryVips: + type: array + description: "An array of Secondary VIPs. All the VIPs will be part of an ipset" + minItems: 1 + items: + type: string + redirectPort: + type: integer + minimum: 1 + maximum: 65535 + description: "Port from which http traffic should be redirected to https" + port: + type: integer + minimum: 1 + maximum: 65535 + certificates: + type: array + description: "certificates attached to the endpoints - Not applicable for HTTP" + minItems: 1 + items: + type: object + properties: + preconfigured: + type: string + description: "Preconfigured Certificate name on NetScaler " + secret: + type: object + description: "Kuberentes secret object" + required: [name] + properties: + name: + type: string + description: "name of the Kubernetes Secret object where Cert is located" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + type: string + description: "Namespace of the kubernetes secret object; Default is same namespace where the Listener object is located" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + default: + type: boolean + description: "Only one of the certificate can be marked as default which will be presented if none of the cert matches with the hostname" + oneOf: + - required: ["preconfigured"] + - required: ["secret"] + policies: + type: object + description: "Policies attached to the Listener" + properties: + httpprofile: + type: object + description: "HTTP profile configurations for the Listener, HTTP level configurations" + properties: + preconfigured: + type: string + description: "Preconfigured or Built-in HTTP profile name" + config: + type: object + description: "HTTP profile configuration for the listener. For individual fields, refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ns/nshttpprofile Name field is auto populated" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + tcpprofile: + type: object + description: "TCP level configurations, uses ns tcpprofile of NetScaler" + properties: + preconfigured: + description: "Preconfigured or Built-in TCP profile name" + type: string + config: + type: object + description: "TCPprofile configurations for the listener. For individual fields refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ns/nstcpprofile ; Name field is auto populated" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + csvserverConfig: + type: object + description: "CS Vserver configuration for the listener" + additionalProperties: + type: string + sslprofile: + type: object + description: "SSL profile configuration" + properties: + preconfigured: + type: string + description: "SSL profile which is preconfigured in NetScaler. Ciphers bound to the profile is not overriden" + config: + description: "NetScaler frontend SSL profile configurations. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/ssl/sslprofile.html for all configurations; Name field is auto generated" + type: object + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + sslciphers: + type: array + description: "List of ciphers to be bound to the ssl profile for the listener. Priority is as per the order in the list. A cipher suite, predefined cipher group or User created cipher group can be mentioned" + minItems: 1 + items: + type: string + description: "Cipher suite, cipher group name" + analyticsprofile: + type: object + description: "Analytics profile configuration" + properties: + preconfigured: + type: array + description: "Preconfigured Analytics profile that needs to be bound to the vserver" + minItems: 1 + items: + type: string + description: "Name of the analytics profile preconfigured that will be bound to the Vserver" + config: + type: array + description: "An array of analytics to be enabled" + minItems: 1 + items: + type: object + description: "Anlytics to be enabled" + required: ['type'] + properties: + type: + description: "Analytics profile to be enabled, you can enable one or more of the webinsight, tcpinsight, securityinsight, videoinsight, hdxinsight, gatewayinsight, timeseries, lsninsight, botinsight " + type: string + enum: ["webinsight", "tcpinsight", "securityinsight", "videoinsight", "hdxinsight", "gatewayinsight", "timeseries", "lsninsight", "botinsight"] + parameters: + type: object + description: "Additional parameters for analytics profile. Please refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/analytics/analyticsprofile/" + additionalProperties: + type: string + oneOf: + - required: ["preconfigured"] + - required: ["config"] + routes: + type: array + description: "List of route objects attached to the listener" + minItems: 1 + items: + type: object + properties: + name: + type: string + description: "Name of the HTTPRoute object" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + namespace: + type: string + description: "Namespace of the HTTPRoute object" + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + labelSelector: + description: "Labels key value pair, if the route carries the same labels, it is automatically attached" + type: object + additionalProperties: + type: string + oneOf: + - required: [name, namespace] + - required: [labelSelector] + defaultAction: + type: object + description: "Default action for the listener: One of Backend or Redirect" + properties: + backend: + type: object + oneOf: + - required: [kube] + properties: + kube: + type: object + required: [service, port] + properties: + service: + description: "Name of the backend service" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + port: + description: "Service port" + type: integer + minimum: 1 + maximum: 65535 + namespace: + description: "Service namespace" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + backendConfig: + description: "General backend service options" + type: object + properties: + secure_backend: + description: "Use Secure communications to the backends" + type: boolean + lbConfig: + description: "NetScaler LB vserver configurations for the backend. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/lb/lbvserver.html for all configurations" + type: object + additionalProperties: + type: string + servicegroupConfig: + description: "NetScaler service group configurations for the backend; Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/basic/servicegroup.html for all configurations" + type: object + additionalProperties: + type: string + redirect: + type: object + oneOf: + - required: [targetExpression] + - required: [hostRedirect] + - required: [httpsRedirect] + properties: + httpsRedirect: + description: "Change the scheme from http to https keeping URL intact" + type: boolean + hostRedirect: + description: "Host name specified is used for redirection with URL intact" + type: string + targetExpression: + description: "A target can be specified using NetScaler policy expression" + type: string + responseCode: + description: "Default response code is 302, which can be customised using this attribute" + type: integer + minimum: 100 + maximum: 599 + oneOf: + - required: ["backend"] + - required: ["redirect"] + subresources: + # status enables the status subresource. + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: httproutes.citrix.com +spec: + group: citrix.com + names: + kind: HTTPRoute + plural: httproutes + singular: httproute + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + required: [rules] + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + hostname: + type: array + description: "List of domain names that share the same route, default is '*'" + minItems: 1 + items: + type: string + description: "Domain name" + rules: + type: array + description: "List Content routing rules with an action defined" + minItems: 1 + items: + type: object + required: [name, action] + properties: + name: + type: string + description: "A name to represent the rule, this is used as an identifier in content routing policy name in NetScaler" + minLength: 1 + maxLength: 20 + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + match: + type: array + description: "List of rules with same action" + minItems: 1 + items: + type: object + anyOf: + - required: [path] + - required: [headers] + - required: [cookies] + - required: [queryParams] + - required: [method] + - required: [policyExpression] + properties: + path: + type: object + description: "URL Path based content routing" + properties: + prefix: + type: string + description: "URL path matches the prefix expression" + exact: + type: string + description: "URL Path must match exact path" + regex: + type: string + description: "PCRE based regex expression for path matching" + headers: + type: array + description: "List of header for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Header details for content routing, Check for existence of a header or header name-value match" + properties: + headerName: + type: object + description: "Header name based content routing, Here existence of header is used for routing" + properties: + exact: + type: string + description: "Header Name - treated as exact must exist" + contains: + type: string + description: "Header Name - A header must exist that contain the string the name" + regex: + type: string + description: "header Name - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e header name must not exist" + oneOf: + - required: [exact] + - required: [contains] + - required: [regex] + headerValue: + type: object + description: "Header Name and Value based match" + properties: + name: + type: string + description: "Header name that must match the value" + exact: + type: string + description: "Header value - treated as exact" + contains: + type: string + description: "Header value - treated as contains" + regex: + type: string + description: "header value - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e header if present must not match the value" + oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + queryParams: + type: array + description: "List of Query parameters for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Query parameters Name and Value based match" + properties: + name: + type: string + description: "Query name that must match the value. If no value is specified, matches with any value" + exact: + type: string + description: "Query value - Exact match" + contains: + type: string + description: "Query value - value must have the string(substring)" + regex: + type: string + description: "Query value - Value must match this regex patterm" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e query if present must not match the value" + anyOf: + - required: [name] + - oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + cookies: + type: array + description: "List of Cookie params for content routing - Must match all the rules- Treated as AND condition if more than 1 rule" + minItems: 1 + items: + type: object + description: "Cookie based routing" + properties: + name: + type: string + description: "cookie name that must match the value. If no value specified, it matches with any value" + exact: + type: string + description: "cookie value - treated as exact" + contains: + type: string + description: "cookie value - treated as substring" + regex: + type: string + description: "cookie value - treated as PCRE regex expression" + not: + type: boolean + description: "Default False, if present, rules are inverted. I.e cookie if present must not match the value" + anyOf: + - required: [name] + - oneOf: + - required: [name, exact] + - required: [name, contains] + - required: [name, regex] + method: + type: string + description: "HTTP method for content routing eg: POST, PUT, DELETE etc" + policyExpression: + type: string + description: "NetScaler policy expressions; refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/policy/policyexpression.html" + action: + type: object + description: "Action for the matched rule" + properties: + backend: + type: object + oneOf: + - required: [kube] + properties: + kube: + type: object + required: [service, port] + properties: + service: + description: "Name of the backend service" + type: string + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + port: + description: "Service port" + type: integer + minimum: 1 + maximum: 65535 + backendConfig: + type: object + description: "General backend service options" + properties: + secureBackend: + description: "Use Secure communications to the backends" + type: boolean + lbConfig: + description: "NetScaler LB vserver configurations for the backend. Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/lb/lbvserver.html for all configurations" + type: object + additionalProperties: + type: string + servicegroupConfig: + description: "NetScaler service group configurations for the backend; Refer: https://developer-docs.netscaler.com/en-us/adc-nitro-api/current-release/configuration/basic/servicegroup.html for all configurations" + type: object + additionalProperties: + type: string + redirect: + type: object + oneOf: + - required: [targetExpression] + - required: [hostRedirect] + - required: [httpsRedirect] + properties: + httpsRedirect: + description: "Change the scheme from http to https keeping URL intact" + type: boolean + hostRedirect: + description: "Host name specified is used for redirection with URL intact" + type: string + targetExpression: + description: "A target can be specified using NetScaler policy expression" + type: string + responseCode: + description: "Default response code is 302, which can be customised using this attribute" + type: integer + minimum: 100 + maximum: 599 + oneOf: + - required: ["backend"] + - required: ["redirect"] + subresources: + # status enables the status subresource. + status: {} + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + # name must match the spec fields below, and be in the form: . + name: continuousdeployments.citrix.com +spec: + group: citrix.com + names: + kind: continuousdeployment + plural: continuousdeployments + singular: continuousdeployment + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + schema: + openAPIV3Schema: + type: object + properties: + spec: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + cronSpec: + type: integer + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: wafs.citrix.com +spec: + group: citrix.com + names: + kind: waf + plural: wafs + singular: waf + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the waf policies are applied.' + type: array + items: + type: string + maxLength: 127 + application_type: + description: 'Type of applications to protect' + type: array + items: + type: string + enum: ['HTML', 'JSON', 'XML'] + signatures: + description: 'Location of external signature file' + type: string + redirect_url: + description: 'When a URL is blocked/down, redirect_url represents the alternate URL where the client requests should be sent.' + type: string + html_error_object: + description: 'Location of customized error page to respond when html or common violation are hit' + type: string + xml_error_object: + description: 'Location of customized error page to respond when xml violations are hit' + type: string + json_error_object: + description: 'Location of customized error page to respond when json violations are hit' + type: string + ip_reputation: + type: object + x-kubernetes-preserve-unknown-fields: true + description: 'Enabling IP reputation feature' + target: + description: 'To control what traffic to be inspected by Web Application Firewall. If you do not provide the target, everything will be inspected by default' + type: object + properties: + path: + type: array + description: "List of http urls to inspect" + items: + type: string + description: "URL path" + method: + type: array + description: "List of http methods to inspect" + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header: + type: array + description: "List of http headers to inspect" + items: + type: string + description: "header name" + security_checks: + description: 'To enable/disable application firewall security checks' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + settings: + description: 'To fine tune application firewall security checks default settings' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + relaxations: + description: 'Section which contains relaxation rules for known traffic and false positives' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true + enforcements: + description: 'Section which contains enforcement or restriction rules' + type: object + properties: + common: + type: object + x-kubernetes-preserve-unknown-fields: true + html: + type: object + x-kubernetes-preserve-unknown-fields: true + json: + type: object + x-kubernetes-preserve-unknown-fields: true + xml: + type: object + x-kubernetes-preserve-unknown-fields: true +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: bots.citrix.com +spec: + group: citrix.com + names: + kind: bot + plural: bots + singular: bot + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'Name of the services to which the bot policies are applied.' + type: array + items: + type: string + maxLength: 127 + signatures: + description: 'Location of external bot signature file' + type: string + redirect_url: + description: 'url to redirect when bot violation is hit' + type: string + target: + description: 'To control what traffic to be inspected by BOT. If you do not provide the target, everything will be inspected by default' + type: object + properties: + path: + type: array + description: "List of http urls to inspect" + items: + type: string + description: "URL path" + method: + type: array + description: "List of http methods to inspect" + items: + type: string + enum: ['GET', 'PUT', 'POST', 'DELETE', 'HEAD', 'OPTIONS', 'TRACE', 'CONNECT','PATCH', 'UNKNOWN_METHOD'] + header: + type: array + description: "List of http headers to inspect" + items: + type: string + description: "header name" + security_checks: + description: 'To enable/disable bot ecurity checks' + type: object + properties: + allow_list: + type: string + enum: ['ON', 'OFF'] + block_list: + type: string + enum: ['ON', 'OFF'] + device_fingerprint: + type: string + enum: ['ON', 'OFF'] + device_fingerprint_action: + type: object + x-kubernetes-preserve-unknown-fields: true + headless_browser: + type: string + enum: ['ON','OFF'] + reputation: + type: string + enum: ['ON', 'OFF'] + ratelimit: + type: string + enum: ['ON', 'OFF'] + tps: + type: string + enum: ['ON', 'OFF'] + trap: + type: object + x-kubernetes-preserve-unknown-fields: true + bindings: + description: 'Section which contains binding rules for bot security checks' + type: object + properties: + allow_list: + type: array + items: + type: object + properties: + subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6_subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + expression: + type: object + x-kubernetes-preserve-unknown-fields: true + + block_list: + type: array + items: + type: object + properties: + subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6: + type: object + x-kubernetes-preserve-unknown-fields: true + ipv6_subnet: + type: object + x-kubernetes-preserve-unknown-fields: true + expression: + type: object + x-kubernetes-preserve-unknown-fields: true + ratelimit: + type: array + items: + type: object + properties: + url: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + cookie: + type: object + x-kubernetes-preserve-unknown-fields: true + geolocation: + type: object + x-kubernetes-preserve-unknown-fields: true + reputation: + type: object + x-kubernetes-preserve-unknown-fields: true + captcha: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + logexp: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + kbmexpr: + type: array + items: + type: object + x-kubernetes-preserve-unknown-fields: true + properties: + tps: + type: object + properties: + geolocation: + type: object + x-kubernetes-preserve-unknown-fields: true + host: + type: object + x-kubernetes-preserve-unknown-fields: true + ip: + type: object + x-kubernetes-preserve-unknown-fields: true + url: + type: object + x-kubernetes-preserve-unknown-fields: true + trapinsertion: + type: object + x-kubernetes-preserve-unknown-fields: true +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: apigatewaypolicies.citrix.com +spec: + group: citrix.com + names: + kind: apigatewaypolicy + plural: apigatewaypolicies + singular: apigatewaypolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + subresources: + status: {} + schema: + openAPIV3Schema: + type: object + required: [spec] + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + api_definition: + type: object + properties: + repository: + type: string + branch: + type: string + oas_secret_ref: + type: string + files: + type: array + items: + type: string + maxLength: 127 + api_proxy: + type: object + properties: + ipaddress: + type: string + port: + type: integer + protocol: + type: string + secret: + type: string + policies: + type: array + items: + type: object + properties: + name: + type: string + selector: + type: array + items: + type: object + properties: + tags: + type: array + items: + type: string + api: + type: string + method: + type: array + items: + type: string + maxLength: 127 + upstream: + type: object + properties: + service: + type: string + port: + type: integer + policy_bindings: + type: object + properties: + ratelimit: + type: object + properties: + name: + type: string + waf: + type: object + properties: + name: + type: string + rewritepolicy: + type: object + properties: + name: + type: string + bot: + type: object + properties: + name: + type: string + aaa: + type: array + items: + type: object + properties: + crd_name: + type: string + mappings: + type: array + items: + type: object + properties: + petstore_auth: + type: string + api_key: + type: string +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: corspolicies.citrix.com +spec: + group: citrix.com + names: + kind: corspolicy + plural: corspolicies + singular: corspolicy + shortNames: + - cp + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: 'Current Status of the CRD' + jsonPath: .status.state + - name: Message + type: string + description: 'Status Message' + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + type: string + description: "Ingress class, if not specified then all NetScaler Ingress Controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + servicenames: + description: 'The list of Kubernetes services to which you want to apply the cors policies.' + type: array + items: + type: string + maxLength: 63 + allow_origin: + description: 'Represents list of allowed origins, it is used to screen the “origin” in the cors pre flight request' + type: array + items: + type: string + maxLength: 2083 + allow_methods: + description: 'Indicates which methods are supported by the response’s URL for the purposes of the CORS protocol. This variable will be used to set Access-Control-Allow-Methods in the pre-flight cors response.' + type: array + items: + type: string + maxLength: 127 + allow_headers: + description: 'Indicates which headers are supported by the response’s URL for the purposes of the CORS protocol. This variable will be used to set Access-Control-Allow-Headers in the pre-flight cors response.' + type: array + items: + type: string + maxLength: 127 + max_age: + description: 'Indicates the number of seconds (5 by default) the information provided by the `Access-Control-Allow-Methods` and `Access-Control-Allow-Headers` headers can be cached. This variable will be used to set Access-Control-Max-Age in the pre-flight cors response.' + type: integer + allow_credentials: + description: 'Indicates whether the response can be shared when the request’s credentials mode is "include". This variable will be set to Access-Control-Allow-Credentials in the rewrite action.' + type: boolean + required: [servicenames, allow_origin, allow_methods, allow_headers] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: appqoepolicies.citrix.com +spec: + group: citrix.com + names: + kind: appqoepolicy + plural: appqoepolicies + singular: appqoepolicy + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + appqoe-policies: + type: array + items: + type: object + properties: + servicenames: + description: 'Name of the services that needs to be binded to appqoe policy.' + type: array + items: + type: string + maxLength: 127 + appqoe-policy: + type: object + properties: + operation-retry: + type: object + properties: + on-reset: + description: "To set Retry on Connection Reset or Not" + type: string + enum: ['YES','NO'] + on-timeout: + description: "Time in milliseconds for retry" + type: integer + minimum: 30 + maximum: 2000 + number-of-retries: + description: "To set number of retries" + type: integer + minimum: 1 + maximum: 7 + required: [operation-retry] + appqoe-criteria: + description: 'Expression against which traffic is evaluated.' + type: string + maxLength: 1299 + direction: + description: 'Bind point to which to bind the policy.' + type: string + enum: ["REQUEST","RESPONSE"] + required: [appqoe-criteria, operation-retry] + required: [appqoe-policy] +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: wildcarddnsentries.citrix.com +spec: + group: citrix.com + names: + kind: wildcarddnsentry + plural: wildcarddnsentries + singular: wildcarddnsentry + scope: Namespaced + versions: + - name: v1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: Current Status of the CRD + jsonPath: .status.state + - name: Message + type: string + description: Status Message + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + zone: + type: object + description: DNS configuration for a zone + properties: + domain: + type: string + description: Domain name + dnsaddrec: + type: object + description: DNS Address record + properties: + domain-ip: + type: string + description: IPv4 addresses to assign to the domain name + ttl: + type: integer + description: >- + TTL is the time for which the record must be cached + by DNS proxies + dnsaaaarec: + type: object + description: DNS AAAA record + properties: + domain-ip: + type: string + description: IPv6 addresses to assign to the domain name + ttl: + type: integer + description: >- + TTL is the time for which the record must be cached + by DNS proxies + soarec: + type: object + description: SOA record + properties: + origin-server: + type: string + description: Origin server domain + contact: + type: string + description: Admin contact + serial: + type: integer + description: >- + The secondary server uses this parameter to + determine whether it requires a zone transfer from + the primary server. + refresh: + type: integer + description: >- + Time, in seconds, for which a secondary server must + wait between successive checks on the value of the + serial number. + retry: + type: integer + description: >- + Time, in seconds, between retries if a secondary server's + attempt to contact the primary server for a zone refresh fails. + expire: + type: integer + description: >- + Time, in seconds, after which the zone data on a secondary + nameserver can no longer be considered authoritative because + all refresh and retry attempts made during the period have failed." + nsrec: + type: object + description: Name server record + properties: + nameserver: + type: string + description: Host name of the name server to add to the domain. + ttl: + type: integer + description: >- + Time to Live (TTL), in seconds, for the record. TTL + is the time for which the record must be cached by + DNS proxies. The specified TTL is applied to all the + resource records that are of the same record type + and belong to the specified domain name +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: icappolicies.citrix.com +spec: + group: citrix.com + names: + kind: icappolicy + plural: icappolicies + singular: icappolicy + scope: Namespaced + versions: + - name: v1beta1 + served: true + storage: true + subresources: + status: {} + additionalPrinterColumns: + - name: Status + type: string + description: "Current Status of the CRD" + jsonPath: .status.state + - name: Message + type: string + description: "Status Message" + jsonPath: .status.status_message + schema: + openAPIV3Schema: + type: object + properties: + status: + type: object + properties: + state: + type: string + status_message: + type: string + spec: + type: object + properties: + ingressclass: + description: "Ingress class, if not specified then all NetScaler ingress controllers in the cluster will process the resource otherwise only the controller with that ingress class will process this resource" + type: string + maxLength: 127 + services: + type: array + description: 'Name of the services for which the icap policy needs to be bound' + items: + type: string + icap-servers: + type: object + description: "ICAP service for the ICAP server that will be part of the load balancing setup. The service that you add provides the ICAP connection between the NetScaler appliance and load balancing virtual servers." + properties: + servers: + type: array + items: + type: object + properties: + ip: + type: string + description: 'IP of the ICAP Server' + format: ipv4 + port: + type: integer + description: 'Port number of the ICAP Server.' + minimum: 1 + maximum: 65535 + required: + - ip + - port + server-type: + type: string + description: 'Type of ICAP Server.' + enum: ['TCP', 'SSL_TCP'] + default: 'SSL_TCP' + server_host_cert: + description: |+ + 'Name of the SSL certificate to be used with ICAP server. + This certificate is mandatory for server-type SSL_TCP' + type: object + properties: + tls_secret: + type: string + description: 'Name of the Kubernetes Secret of type tls referring to Certificate' + pattern: '^[a-z0-9]([-a-z0-9]*[a-z0-9])?$' + preconfigured: + type: string + maxLength: 63 + description: |+ + 'Preconfigured SSL certkey name on NetScaler with the + certificate and key already added on NetScaler' + oneOf: + - required: [tls_secret] + - required: [preconfigured] + required: + - servers + icap: + type: array + items: + type: object + properties: + preconfigured-profile: + description: 'Names of the preconfigured ICAP profile.' + type: string + maxLength: 127 + direction: + description: 'ICAP Mode of operation. It is a mandatory argument while creating an icapprofile.' + type: string + enum: ['REQUEST','RESPONSE'] + profile: + type: object + description: 'ICAP profile(s) of the NetScaler.' + properties: + preview: + description: 'Enable or Disable preview header with ICAP request. This feature allows an ICAP server to see the beginning of a transaction, then decide if it wants to opt-out of the transaction early instead of receiving the remainder of the request message.' + type: string + enum: ["ENABLED", "DISABLED"] + preview-length: + description: 'Value of Preview Header field. NetScaler uses the minimum of this set value and the preview size received on OPTIONS' + type: integer + minimum: 0 + maximum: 4294967294 + uri: + description: 'URI representing icap service. It is a mandatory argument while creating an icapprofile.' + type: string + maxLength: 511 + host-header: + description: 'ICAP Host Header.' + type: string + maxLength: 255 + user-agent-header: + description: 'ICAP User Agent Header' + type: string + maxLength: 255 + query-params: + description: 'Query parameters to be included with ICAP request URI. Entered values should be in arg=value format. For more than one parameters, add & separated values. e.g.: arg1=val1&arg2=val2' + type: string + maxLength: 511 + connection-keep-alive: + description: 'Enable or Disable sending Allow: 204 header in ICAP request.' + type: string + enum: ["ENABLED", "DISABLED"] + insert-icap-headers: + description: 'Insert custom ICAP headers in the ICAP request to send to ICAP server. The headers can be static or can be dynamically constructed using PI Policy Expression. For example, to send static user agent and Client''s IP address, the expression can be specified as "User-Agent: NS-ICAP-Client/V1.0r0-Client-IP: "+CLIENT.IP.SRC+"r0. The NetScaler does not check the validity of the specified header name-value. You must manually validate the specified header syntax.' + type: string + maxLength: 8191 + insert-http-request: + description: 'Exact HTTP request, in the form of an expression, which the NetScaler encapsulates and sends to the ICAP server. If you set this parameter, the ICAP request is sent using only this header. This can be used when the HTTP header is not available to send or ICAP server only needs part of the incoming HTTP request. The request expression is constrained by the feature for which it is used. The NetScaler does not check the validity of this request. You must manually validate the request.' + type: string + maxLength: 8191 + req-timeout: + description: 'Time, in seconds, within which the remote server should respond to the ICAP-request. If the Netscaler does not receive full response with this time, the specified request timeout action is performed. Zero value disables this timeout functionality.' + type: integer + minimum: 0 + maximum: 86400 + req-timeout-action: + description: 'Name of the action to perform if the Vserver/Server representing the remote service does not respond with any response within the timeout value configured. The Supported actions are * BYPASS - This Ignores the remote server response and sends the request/response to Client/Server. * If the ICAP response with Encapsulated headers is not received within the request-timeout value configured, this Ignores the remote ICAP server response and sends the Full request/response to Server/Client' + type: string + enum: ['BYPASS', 'DROP', 'RESET'] + log-action: + description: 'Name of the audit message action which would be evaluated on receiving the ICAP response to emit the logs' + type: string + maxLength: 127 + required: + - uri + content-inspection-criteria: + description: 'Expression that the policy uses to determine whether to execute the specified action.' + type: string + maxLength: 1499 + default-action: + description: 'Action to perform if the result of policy evaluation is undefined (UNDEF). An UNDEF event indicates an internal error condition. Only the above built-in actions can be used' + type: string + maxLength: 127 + log-action: + description: 'Name of the messagelog action to use for requests that match this policy.' + type: string + maxLength: 127 + goto-priority-expression: + description: 'Expression or other value specifying the next policy to be evaluated if the current policy evaluates to TRUE.Specify one of the following values:* NEXT - Evaluate the policy with the next higher priority number.* END - End policy evaluation.Default value of goto-priority-expression: END' + type: string + operation: + description: 'Type of operation this action is going to perform. following actions are available to configure: * ICAP - forward the incoming request or response to an ICAP server for modification. * INLINEINSPECTION - forward the incoming or outgoing packets to IPS server for Intrusion Prevention. * MIRROR - Forwards cloned packets for Intrusion Detection. * NOINSPECTION - This does not forward incoming and outgoing packets to the Inspection device. * NSTRACE - capture current and further incoming packets on this transaction.' + type: string + enum: ['ICAP', 'INLINEINSPECTION', 'MIRROR', 'NOINSPECTION'] + server-failure-action: + description: 'Name of the action to perform if the Vserver representing the remote service is not UP. This is not supported for NOINSPECTION Type. The Supported actions are: * RESET - Reset the client connection by closing it. The client program, such as a browser, will handle this and may inform the user. The client may then resend the request if desired. * DROP - Drop the request without sending a response to the user. * CONTINUE - It bypasses the ContentIsnpection and Continues/resumes the Traffic-Flow to Client/Server.' + type: string + enum: ['CONTINUE', 'DROP', 'RESET'] + oneOf: + - required: [preconfigured-profile] + - required: [profile] + required: + - direction + - content-inspection-criteria + - operation + required: + - ingressclass + - services + - icap-servers + - icap +--- diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/NOTES.txt b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/NOTES.txt new file mode 100644 index 000000000..1bab5db5c --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/NOTES.txt @@ -0,0 +1,15 @@ +Thank you for installing {{ .Chart.Name }}. + +Your release is named {{ .Release.Name }}. + + +To learn more about the release, try: + + $ helm status {{ .Release.Name }} + $ helm get {{ .Release.Name }} + + +To delete : + helm delete {{ .Release.Name }} + + diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/_helpers.tpl b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/_helpers.tpl new file mode 100644 index 000000000..4f3942e48 --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/_helpers.tpl @@ -0,0 +1,79 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "netscaler-ingress-controller.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 "netscaler-ingress-controller.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 -}} + +{{- define "exporter.fullname" -}} +{{- $name := default .Chart.Name "exporter" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "servicemonitor.fullname" -}} +{{- $name := default .Chart.Name "netscaler-adc-servicemonitor" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "servicemonitorlabel" -}} +{{- $name := default .Chart.Name "netscaler-adc-svcmon" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{- define "nsicconfigmap.fullname" -}} +{{- $name := default .Chart.Name "nsic-configmap" .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "netscaler-ingress-controller.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account to use +*/}} +{{- define "netscaler-ingress-controller.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "netscaler-ingress-controller.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/configmap.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/configmap.yaml new file mode 100644 index 000000000..0a721801b --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/configmap.yaml @@ -0,0 +1,82 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "nsicconfigmap.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + LOGLEVEL: {{ .Values.logLevel | quote | lower }} + JSONLOG: {{ .Values.jsonLog | quote | lower }} + NS_PROTOCOL: {{ .Values.nsProtocol | quote | lower }} + NS_PORT: {{ .Values.nsPort | quote }} +{{- if .Values.nsSNIPS }} + NS_SNIPS: {{ .Values.nsSNIPS | toJson}} +{{- end }} +{{- if and .Values.analyticsConfig.required .Values.nsEnableLabel }} + NS_ENABLE_LABELS: {{ .Values.nsEnableLabel | quote}} +{{- end }} +{{- if .Values.podIPsforServiceGroupMembers }} + POD_IPS_FOR_SERVICEGROUP_MEMBERS: {{ .Values.podIPsforServiceGroupMembers | quote }} +{{- end }} +{{- if .Values.ignoreNodeExternalIP }} + IGNORE_NODE_EXTERNAL_IP: {{ .Values.ignoreNodeExternalIP | quote }} +{{- end }} + +{{- if ne (upper .Values.nsHTTP2ServerSide) "OFF" }} + NS_HTTP2_SERVER_SIDE: {{ .Values.nsHTTP2ServerSide | quote }} +{{- end }} +{{- if ne (toString .Values.nsCookieVersion) "0" }} + NS_COOKIE_VERSION: {{ .Values.nsCookieVersion | quote }} +{{- end }} +{{- if .Values.nsDnsNameserver }} + NS_DNS_NAMESERVER: {{ .Values.nsDnsNameserver }} +{{- end }} + +{{- if .Values.analyticsConfig.required }} + NS_ANALYTICS_CONFIG: | + distributed_tracing: + enable: {{ .Values.analyticsConfig.distributedTracing.enable | quote }} + samplingrate: {{ .Values.analyticsConfig.distributedTracing.samplingrate }} + endpoint: + {{- if not .Values.analyticsConfig.timeseries.metrics.enableNativeScrape }} + metrics: + service: {{ .Values.analyticsConfig.endpoint.metrics.service | quote }} + {{- end }} + transactions: + service: {{ .Values.analyticsConfig.endpoint.transactions.service | quote }} + timeseries: + port: {{ .Values.analyticsConfig.timeseries.port }} + metrics: + enable: {{ .Values.analyticsConfig.timeseries.metrics.enable | quote }} + mode: {{ .Values.analyticsConfig.timeseries.metrics.mode | quote }} + export_frequency: {{ .Values.analyticsConfig.timeseries.metrics.exportFrequency }} + schema_file: {{ .Values.analyticsConfig.timeseries.metrics.schemaFile | quote }} + enable_native_scrape: {{ .Values.analyticsConfig.timeseries.metrics.enableNativeScrape | quote }} + auditlogs: + enable: {{ .Values.analyticsConfig.timeseries.auditlogs.enable | quote }} + events: + enable: {{ .Values.analyticsConfig.timeseries.events.enable | quote }} + transactions: + enable: {{ .Values.analyticsConfig.transactions.enable | quote }} + port: {{ .Values.analyticsConfig.transactions.port }} +{{- end }} + +{{- if .Values.nsLbHashAlgo.required }} + NS_LB_HASH_ALGO: | + hashFingers: {{ .Values.nsLbHashAlgo.hashFingers }} + hashAlgorithm: {{ .Values.nsLbHashAlgo.hashAlgorithm | quote }} +{{- end }} + +{{- if .Values.profileSslFrontend }} + FRONTEND_SSL_PROFILE: | + {{- toYaml .Values.profileSslFrontend | nindent 4 }} +{{- end }} + +{{- if .Values.profileTcpFrontend }} + FRONTEND_TCP_PROFILE: | + {{- toYaml .Values.profileTcpFrontend | nindent 4 }} +{{- end }} + +{{- if .Values.profileHttpFrontend }} + FRONTEND_HTTP_PROFILE: | + {{- toYaml .Values.profileHttpFrontend | nindent 4 }} +{{- end }} diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/deployment.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/deployment.yaml new file mode 100644 index 000000000..68513bbbd --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/deployment.yaml @@ -0,0 +1,282 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "netscaler-ingress-controller.fullname" . }} + namespace: {{ .Release.Namespace }} +spec: + selector: + matchLabels: +{{- if .Values.openshift }} + router: {{ include "netscaler-ingress-controller.fullname" . }} +{{- else }} + app: {{ include "netscaler-ingress-controller.fullname" . }} +{{- end }} + replicas: 1 + template: + metadata: + name: nsic + labels: +{{- if .Values.openshift }} + router: {{ include "netscaler-ingress-controller.fullname" . }} +{{- else }} + app: {{ include "netscaler-ingress-controller.fullname" . }} +{{- end }} + annotations: +{{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} +{{- end }} + spec: + serviceAccountName: {{ include "netscaler-ingress-controller.serviceAccountName" . }} + containers: + - name: nsic + image: "{{ tpl .Values.image . }}" + imagePullPolicy: {{ .Values.pullPolicy }} +{{- if .Values.enableReadinessProbe }} + readinessProbe: + {{- toYaml .Values.readinessProbe | nindent 10 }} +{{- end }} +{{- if .Values.enableLivenessProbe }} + livenessProbe: + {{- toYaml .Values.livenessProbe | nindent 10 }} +{{- end }} + args: + - --configmap + {{ .Release.Namespace }}/{{ include "nsicconfigmap.fullname" . }} +{{- if .Values.defaultSSLCertSecret }} + - --default-ssl-certificate + {{ .Release.Namespace }}/{{ .Values.defaultSSLCertSecret }} +{{- end }} +{{- if .Values.defaultSSLSNICertSecret }} + - --default-ssl-sni-certificate + {{ .Release.Namespace }}/{{ .Values.defaultSSLSNICertSecret }} +{{- end }} +{{- if .Values.ingressClass }} + - --ingress-classes +{{- range .Values.ingressClass}} + {{.}} +{{- end }} +{{- end }} +{{- if .Values.serviceClass }} + - --service-classes +{{- range .Values.serviceClass}} + {{.}} +{{- end }} +{{- end }} + - --feature-node-watch + {{ .Values.nodeWatch }} + - --enable-cnc-pbr + {{ .Values.nsncPbr }} +{{- if .Values.ipam }} + - --ipam + citrix-ipam-controller +{{- end }} +{{- if .Values.disableAPIServerCertVerify }} + - --disable-apiserver-cert-verify + {{ .Values.disableAPIServerCertVerify }} +{{- end }} +{{- if .Values.updateIngressStatus }} + - --update-ingress-status + yes +{{- end }} + env: + - name: "NS_IP" + value: "{{ .Values.nsIP }}" +{{- if .Values.nsVIP }} + - name: "NS_VIP" + value: "{{ .Values.nsVIP }}" +{{- end }} +{{- if .Values.rbacRole }} + - name: "SCOPE" + value: "local" +{{- end }} +{{- if .Values.nitroReadTimeout }} + - name: "NS_NITRO_READ_TIMEOUT" + value: "{{ .Values.nitroReadTimeout }}" +{{- end }} +{{- if .Values.enableLivenessProbe }} + - name: "LIVENESS_FILE_PATH" + value: '/tmp/liveness_path.log' +{{- end }} + - name: "ENABLE_LIVENESS_PROBE" + value: {{ .Values.enableLivenessProbe | quote }} + - name: "NS_USER" + {{- if and .Values.secretStore.enabled .Values.secretStore.username}} + {{- toYaml .Values.secretStore.username | nindent 10 }} + {{- else }} + valueFrom: + secretKeyRef: + name: {{ .Values.adcCredentialSecret }} + key: username + {{- end }} + - name: "NS_PASSWORD" + {{- if and .Values.secretStore.enabled .Values.secretStore.password}} + {{- toYaml .Values.secretStore.password | nindent 10 }} + {{- else }} + valueFrom: + secretKeyRef: + name: {{ .Values.adcCredentialSecret }} + key: password + {{- end }} + - name: "EULA" + value: "{{ .Values.license.accept }}" +{{- if and .Values.openshift .Values.routeLabels }} + - name: "ROUTE_LABELS" + value: {{ .Values.routeLabels | quote}} +{{- end }} +{{- if and .Values.openshift .Values.namespaceLabels }} + - name: "NAMESPACE_LABELS" + value: {{ .Values.namespaceLabels | quote }} +{{- end }} + - name: "NS_APPS_NAME_PREFIX" + value: {{ .Values.entityPrefix | default "k8s"| quote }} + - name: "NS_MC_PREFIX" + value: {{ .Values.multiClusterPrefix | default "mc"| quote }} +{{- if .Values.kubernetesURL }} + - name: "kubernetes_url" + value: "{{ .Values.kubernetesURL }}" +{{- end }} +{{- if .Values.clusterName }} + - name: "CLUSTER_NAME" + value: "{{ .Values.clusterName }}" +{{- end }} +{{- if .Values.logProxy }} + - name: "NS_LOGPROXY" + value: "{{ .Values.logProxy }}" +{{- end }} +{{- if .Values.disableOpenshiftRoutes }} + - name: "DISABLE_OPENSHIFT_ROUTES" + value: "{{ .Values.disableOpenshiftRoutes }}" +{{- end }} +{{- if .Values.nsConfigDnsRec }} + - name: "NS_CONFIG_DNS_REC" + value: "{{ .Values.nsConfigDnsRec }}" +{{- end }} +{{- if .Values.nsSvcLbDnsRec }} + - name: "NS_SVC_LB_DNS_REC" + value: "{{ .Values.nsSvcLbDnsRec }}" +{{- end }} +{{- if .Values.optimizeEndpointBinding }} + - name: "OPTIMIZE_ENDPOINT_BINDING" + value: "{{ .Values.optimizeEndpointBinding }}" +{{- end }} +{{- if .Values.nodeLabels }} + - name: "NODE_LABELS" + value: "{{ .Values.nodeLabels }}" +{{- end }} +{{- if .Values.openshift }} + - name: "PLATFORM" + value: "OPENSHIFT" +{{- else }} + - name: "PLATFORM" + value: "KUBERNETES" +{{- end }} + - name: "BGP_ADVERTISEMENT" + value: {{ .Values.bgpAdvertisement | quote }} + resources: + {{- toYaml .Values.resources | nindent 12 }} + {{- if ne (len .Values.extraVolumeMounts) 0 }} + volumeMounts: +{{- toYaml .Values.extraVolumeMounts | nindent 8 }} + {{- end }} +{{- if .Values.exporter.required }} + - name: exporter + image: "{{ tpl .Values.exporter.image . }}" + imagePullPolicy: {{ .Values.exporter.pullPolicy }} + args: + - "--target-nsip={{ .Values.nsIP }}" + - "--port={{ .Values.exporter.ports.containerPort }}" + env: + - name: "NS_USER" + {{- if and .Values.secretStore.enabled .Values.secretStore.username}} + {{- toYaml .Values.secretStore.username | nindent 10 }} + {{- else }} + valueFrom: + secretKeyRef: + name: {{ .Values.adcCredentialSecret }} + key: username + {{- end }} + - name: "NS_PASSWORD" + {{- if and .Values.secretStore.enabled .Values.secretStore.password}} + {{- toYaml .Values.secretStore.password | nindent 10 }} + {{- else }} + valueFrom: + secretKeyRef: + name: {{ .Values.adcCredentialSecret }} + key: password + {{- end }} + {{- if ne (len .Values.exporter.extraVolumeMounts) 0 }} + volumeMounts: + {{- toYaml .Values.exporter.extraVolumeMounts | nindent 8 }} + {{- end }} + resources: +{{- toYaml .Values.exporter.resources | nindent 12 }} +{{- end }} +{{- if or (and .Values.extraVolumeMounts .Values.extraVolumes) (and .Values.exporter.extraVolumeMounts .Values.extraVolumes) }} + volumes: +{{- end }} +{{- if ne (len .Values.extraVolumes) 0 }} +{{ toYaml .Values.extraVolumes | indent 6 }} +{{- end }} +{{- if and .Values.nodeSelector.key .Values.nodeSelector.value }} + nodeSelector: + {{ .Values.nodeSelector.key }}: {{ .Values.nodeSelector.value }} +{{- end }} +{{- if .Values.tolerations }} + tolerations: {{ .Values.tolerations | toYaml | nindent 8 }} +{{- end }} +{{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} +{{- end }} + +--- + +{{- if .Values.exporter.required }} + + +apiVersion: v1 +kind: Service +metadata: + name: {{ include "exporter.fullname" . }} + labels: + app: {{ include "exporter.fullname" . }} + service-type: {{ include "servicemonitorlabel" . }} +spec: + type: ClusterIP + ports: + - port: {{ .Values.exporter.ports.containerPort }} + targetPort: {{ .Values.exporter.ports.containerPort }} + name: exporter-port + selector: +{{- if .Values.openshift }} + router: {{ include "netscaler-ingress-controller.fullname" . }} +{{- else }} + app: {{ include "netscaler-ingress-controller.fullname" . }} +{{- end }} + +--- + +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "servicemonitor.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + servicemonitor: netscaler + {{- with .Values.exporter.serviceMonitorExtraLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + endpoints: + - interval: 30s + port: exporter-port + selector: + matchLabels: + service-type: {{ include "servicemonitorlabel" . }} + namespaceSelector: + matchNames: + - monitoring + - default + - {{ .Release.Namespace }} + +{{- end }} diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/ingressclass.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/ingressclass.yaml new file mode 100644 index 000000000..da86715cc --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/ingressclass.yaml @@ -0,0 +1,18 @@ +{{- $default := .Values.setAsDefaultIngressClass -}} +{{- if semverCompare ">=1.19.0-0" .Capabilities.KubeVersion.GitVersion }} +{{- if .Values.ingressClass }} +{{- range .Values.ingressClass }} +apiVersion: networking.k8s.io/v1 +kind: IngressClass +metadata: + name: {{ . | quote }} +{{- if $default }} + annotations: + ingressclass.kubernetes.io/is-default-class: "true" +{{- end }} +spec: + controller: citrix.com/ingress-controller +--- +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/rbac.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/rbac.yaml new file mode 100644 index 000000000..42dd7e9d7 --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/templates/rbac.yaml @@ -0,0 +1,106 @@ +{{- if not .Values.rbacRole }} +kind: ClusterRole +{{- else }} +kind: Role +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "netscaler-ingress-controller.serviceAccountName" . }} +{{- if .Values.rbacRole }} + namespace: {{ .Release.Namespace }} +{{- end }} +rules: + - apiGroups: [""] +{{- if .Values.openshift }} + resources: ["endpoints", "pods", "secrets", "routes", "tokenreviews", "subjectaccessreviews", "nodes", "namespaces", "configmaps", "services"] +{{- else }} + resources: ["endpoints", "pods", "secrets", "routes", "nodes", "namespaces", "configmaps", "services"] +{{- end }} + verbs: ["get", "list", "watch"] + # services/status is needed to update the loadbalancer IP in service status for integrating + # service of type LoadBalancer with external-dns + - apiGroups: [""] + resources: ["services/status"] + verbs: ["patch"] + - apiGroups: [""] + resources: ["events"] + verbs: ["create"] + - apiGroups: ["extensions", "networking.k8s.io"] + resources: ["ingresses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["extensions","networking.k8s.io"] + resources: ["ingresses/status"] + verbs: ["patch"] + - apiGroups: ["networking.k8s.io"] + resources: ["ingressclasses"] + verbs: ["get", "list", "watch"] + - apiGroups: ["apiextensions.k8s.io"] + resources: ["customresourcedefinitions"] + verbs: ["get", "list", "watch"] + - apiGroups: ["apps"] + resources: ["deployments"] + verbs: ["get", "list", "watch"] + - apiGroups: ["citrix.com"] + resources: ["rewritepolicies", "icappolicies", "continuousdeployments", "authpolicies", "ratelimits", "listeners", "httproutes", "wafs", "apigatewaypolicies", "bots", "corspolicies", "appqoepolicies", "wildcarddnsentries"] + verbs: ["get", "list", "watch", "create", "delete", "patch"] + - apiGroups: ["citrix.com"] + resources: ["rewritepolicies/status", "icappolicies/status", "continuousdeployments/status", "authpolicies/status", "ratelimits/status", "listeners/status", "httproutes/status", "wafs/status", "apigatewaypolicies/status", "bots/status", "corspolicies/status", "appqoepolicies/status", "wildcarddnsentries/status"] + verbs: ["patch"] + - apiGroups: ["citrix.com"] + resources: ["vips"] + verbs: ["get", "list", "watch", "create", "delete"] + - apiGroups: ["crd.projectcalico.org"] + resources: ["ipamblocks"] + verbs: ["get", "list", "watch"] +{{- if .Values.openshift }} + - apiGroups: ["route.openshift.io"] + resources: ["routes"] + verbs: ["get", "list", "watch"] + - apiGroups: ["network.openshift.io"] + resources: ["hostsubnets"] + verbs: ["get", "list", "watch"] + - apiGroups: ["config.openshift.io"] + resources: ["networks"] + verbs: ["get", "list"] +{{- end }} + +--- + +{{- if not .Values.rbacRole }} +kind: ClusterRoleBinding +{{- else }} +kind: RoleBinding +{{- end }} +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "netscaler-ingress-controller.serviceAccountName" . }} +{{- if .Values.rbacRole }} + namespace: {{ .Release.Namespace }} +{{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io +{{- if not .Values.rbacRole }} + kind: ClusterRole +{{- else }} + kind: Role +{{- end }} + name: {{ include "netscaler-ingress-controller.serviceAccountName" . }} +subjects: +- kind: ServiceAccount + name: {{ include "netscaler-ingress-controller.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + +--- + +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "netscaler-ingress-controller.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- if .Values.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.imagePullSecrets }} +- name: {{.}} +{{- end }} +{{- end }} + diff --git a/charts/netscaler/netscaler-ingress-controller/2.2.10/values.yaml b/charts/netscaler/netscaler-ingress-controller/2.2.10/values.yaml new file mode 100644 index 000000000..4f6c0f59e --- /dev/null +++ b/charts/netscaler/netscaler-ingress-controller/2.2.10/values.yaml @@ -0,0 +1,230 @@ +# Default values for netscaler-ingress-controller. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# NetScaler Ingress Controller config details +imageRegistry: quay.io +imageRepository: netscaler/netscaler-k8s-ingress-controller +imageTag: 2.2.10 +image: "{{ .Values.imageRegistry }}/{{ .Values.imageRepository }}:{{ .Values.imageTag }}" +pullPolicy: IfNotPresent +imagePullSecrets: [] +nameOverride: "" +fullnameOverride: "" +openshift: false +adcCredentialSecret: "" # K8s Secret Name +# Enable secretStore to implement CSI Secret Provider classes for holding the nslogin credentials +secretStore: + enabled: false + username: {} + #valueFrom: + # configMapKeyRef: + # name: test1 + # key: username + password: {} + #valueFrom: + # configMapKeyRef: + # name: test1 + # key: password +nsIP: "" +nsVIP: "" +nsSNIPS: [] +license: + accept: no +nsPort: 443 +nsProtocol: HTTPS +nsEnableLabel: true +# nitroReadTimeout is timeout value in seconds for nitro api read timeout(default is 20) +nitroReadTimeout: 20 +logLevel: INFO +jsonLog: false +multiClusterPrefix: +entityPrefix: "" +kubernetesURL: "" +clusterName: "" +ingressClass: [] +setAsDefaultIngressClass: False +serviceClass: [] +defaultSSLCertSecret: "" +defaultSSLSNICertSecret: "" +podIPsforServiceGroupMembers: False +ignoreNodeExternalIP: False +ipam: False +# API server Cert verification can be disabled, while communicating with API Server, if disableAPIServerCertVerify set to True +disableAPIServerCertVerify: False +logProxy: "" +nodeWatch: false +nsncPbr: False +nodeSelector: + key: "" + value: "" +tolerations: [] +updateIngressStatus: True +nsHTTP2ServerSide: "OFF" +nsCookieVersion: "0" +nsConfigDnsRec: False +nsSvcLbDnsRec: False +nsDnsNameserver: "" +optimizeEndpointBinding: False +routeLabels: "" +namespaceLabels: "" +disableOpenshiftRoutes: False +profileSslFrontend: {} + # preconfigured: my_ssl_profile + # OR + # config: + # tls13: 'ENABLED' + # hsts: 'ENABLED' +profileHttpFrontend: {} + # preconfigured: my_http_profile + # OR + # config: + # dropinvalreqs: 'ENABLED' + # websocket: 'ENABLED' +profileTcpFrontend: {} + # preconfigured: my_tcp_profile + # OR + # config: + # sack: 'ENABLED' + # nagle: 'ENABLED' + +# Exporter config details +exporter: + required: false + imageRegistry: quay.io + imageRepository: netscaler/netscaler-adc-metrics-exporter + imageTag: 1.4.9 + image: "{{ .Values.exporter.imageRegistry }}/{{ .Values.exporter.imageRepository }}:{{ .Values.exporter.imageTag }}" + pullPolicy: IfNotPresent + ports: + containerPort: 8888 + 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 + extraVolumeMounts: [] + # We usually recommend not to specify default resources and to leave this as a conscious + # choice for the user. + #- name: github-key + # mountPath: /etc/config/keys/ + # readOnly: true + #- name: agent-init-scripts + # mountPath: /docker-entrypoint.d/ + + serviceMonitorExtraLabels: {} + +# Enable RBAC role (so called local role), by default NSIC deployed with ClusterRole. +# below variable to deploy NSIC with RBAC role, only ingress service supported with this config +rbacRole: False + +# Config required to be done by NetScaler Ingress Controller for sending metrics to NetScaler Observability Exporter +analyticsConfig: + required: false + distributedTracing: + enable: false + samplingrate: 100 + endpoint: + metrics: + service: "" + transactions: + service: "" + timeseries: + port: 30002 + metrics: + enable: false + mode: 'avro' + exportFrequency: 30 + schemaFile: schema.json + enableNativeScrape: false + auditlogs: + enable: false + events: + enable: false + transactions: + enable: false + port: 30001 + +nsLbHashAlgo: + required: false + hashFingers: 256 + hashAlgorithm: 'DEFAULT' + +# Specifies whether a ServiceAccount should be created +serviceAccount: + 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: + +podAnnotations: {} + +resources: + requests: + cpu: 32m + memory: 128Mi + # Following values depends on no of ingresses configured by Ingress Controllers, so it is + # advised to test with maximum no of ingresses to set these values. + # limits: + # cpu: 1000m + # memory: 1000Mi + limits: {} + # Following values depends on no of ingresses configured by Ingress Controllers, so it is + # advised to test with maximum no of ingresses to set these values. + # limits: + # cpu: 1000m + # memory: 1000Mi + +affinity: {} + +bgpAdvertisement: False +nodeLabels: "" +enableReadinessProbe: True +readinessProbe: + exec: + command: + - cat + - /tmp/readiness + initialDelaySeconds: 10 + periodSeconds: 60 + failureThreshold: 3 + successThreshold: 1 + +enableLivenessProbe: True +livenessProbe: + exec: + command: + - /bin/sh + - -c + - | + FILE_PATH="$LIVENESS_FILE_PATH" + [ -f "$FILE_PATH" ] && [ $(( $(date +%s) - $(stat -c %Y "$FILE_PATH") )) -lt 60 ] && exit 0 || exit 1 + initialDelaySeconds: 30 + periodSeconds: 60 + +extraVolumeMounts: [] + # We usually recommend not to specify default resources and to leave this as a conscious + # choice for the user. + #- name: github-key + # mountPath: /etc/config/keys/ + # readOnly: true + #- name: agent-init-scripts + # mountPath: /docker-entrypoint.d/ + +extraVolumes: [] + # We usually recommend not to specify default resources and to leave this as a conscious + # choice for the user. + #- name: agent-init-scripts + # configMap: + # name: agent-init-scripts + # defaultMode: 0755 + #- name: github-key + # secret: + # secretName: github-key + # defaultMode: 0744 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/.helmignore b/charts/trilio/k8s-triliovault-operator/5.0.0/.helmignore new file mode 100644 index 000000000..be86b789d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/.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 +# Helm files +OWNERS diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/BUILD.bazel b/charts/trilio/k8s-triliovault-operator/5.0.0/BUILD.bazel new file mode 100644 index 000000000..c578eb034 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/BUILD.bazel @@ -0,0 +1,9 @@ +load("@bazel_tools//tools/build_defs/pkg:pkg.bzl", "pkg_tar") + +pkg_tar( + name = "helm-tar", + files = glob(["**"]), + package_dir = "/opt/tvk/k8s-triliovault-operator/", + strip_prefix = "./", + visibility = ["//visibility:public"], +) diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/Chart.yaml new file mode 100644 index 000000000..99ed11c1f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/Chart.yaml @@ -0,0 +1,24 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: TrilioVault for Kubernetes Operator + catalog.cattle.io/kube-version: '>=1.19.0-0' + catalog.cattle.io/release-name: k8s-triliovault-operator +apiVersion: v2 +appVersion: 5.0.0 +dependencies: +- condition: observability.enabled + name: observability + repository: file://charts/observability + version: ^0.1.0 +description: K8s-TrilioVault-Operator is an operator designed to manage the K8s-TrilioVault + Application Lifecycle. +home: https://github.com/trilioData/k8s-triliovault-operator +icon: file://assets/icons/k8s-triliovault-operator.png +kubeVersion: '>=1.19.0-0' +maintainers: +- email: prafull.ladha@trilio.io + name: prafull11 +name: k8s-triliovault-operator +sources: +- https://github.com/trilioData/k8s-triliovault-operator +version: 5.0.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/LICENSE b/charts/trilio/k8s-triliovault-operator/5.0.0/LICENSE new file mode 100644 index 000000000..76b559d3b --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/LICENSE @@ -0,0 +1 @@ +# Placeholder for the License if we decide to provide one diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/README.md b/charts/trilio/k8s-triliovault-operator/5.0.0/README.md new file mode 100644 index 000000000..1c8cb3841 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/README.md @@ -0,0 +1,206 @@ +# K8s-TrilioVault-Operator +This operator is to manage the lifecycle of TrilioVault Backup/Recovery solution. This operator install, updates and manage the TrilioVault application. + +## Introduction + +## Prerequisites + +- Kubernetes 1.19+ +- PV provisioner support +- CSI driver should be installed + +### One Click Installation + +In one click install for upstream operator, a cluster scope TVM custom resource `triliovault-manager` is created. + +```shell script +helm repo add trilio-vault-operator https://charts.k8strilio.net/trilio-stable/k8s-triliovault-operator +helm install tvm trilio-vault-operator/k8s-triliovault-operator +``` + +#### One click install with preflight Configuration + +The following table lists the configuration parameter of the upstream operator one click install feature as well as preflight check flags, their default values and usage. + +| Parameter | Description | Default | Example | +|--------------------------------------------------------------------|---------------------------------------------------------------------------------------------------|------------|-------------------------| +| `installTVK.enabled` | 1 click install feature is enabled | true | | +| `installTVK.applicationScope` | scope of TVK application created | Cluster | | +| `installTVK.tvkInstanceName` | tvk instance name | "" | "tvk-instance" | +| `installTVK.ingressConfig.host` | host of the ingress resource created | "" | | +| `installTVK.ingressConfig.tlsSecretName` | tls secret name which contains ingress certs | "" | | +| `installTVK.ingressConfig.annotations` | annotations to be added on ingress resource | "" | | +| `installTVK.ingressConfig.ingressClass` | ingress class name for the ingress resource | "" | | +| `installTVK.ComponentConfiguration.ingressController.enabled` | TVK ingress controller should be deployed | true | | +| `installTVK.ComponentConfiguration.ingressController.service.type` | TVK ingress controller service type | "NodePort" | | +| `preflight.enabled` | enables preflight check for tvk | false | | +| `preflight.storageClass` | Name of storage class to use for preflight checks (Required) | "" | | +| `preflight.cleanupOnFailure` | Cleanup the resources on cluster if preflight checks fail (Optional) | false | | +| `preflight.imagePullSecret` | Name of the secret for authentication while pulling the images from the local registry (Optional) | "" | | +| `preflight.limits` | Pod memory and cpu resource limits for DNS and volume snapshot preflight check (Optional) | "" | "cpu=600m,memory=256Mi" | +| `preflight.localRegistry` | Name of the local registry from where the images will be pulled (Optional) | "" | | +| `preflight.nodeSelector` | Node selector labels for pods to schedule on a specific nodes of cluster (Optional) | "" | "key=value" | +| `preflight.pvcStorageRequest` | PVC storage request for volume snapshot preflight check (Optional) | "" | "2Gi" | +| `preflight.requests` | Pod memory and cpu resource requests for DNS and volume snapshot preflight check (Optional) | "" | "cpu=300m,memory=128Mi" | +| `preflight.volumeSnapshotClass` | Name of volume snapshot class to use for preflight checks (Optional) | "" | | +| `preflight.logLevel` | Log Level for the preflight run (Default: "INFO") | "" | | +| `preflight.imageTag` | Image tag to use for the preflight image (Default: latest) | "" | | +| `nodeSelector` | Node selection constraints for scheduling Pods of this application. | {} | | +| `affinity` | Affinity rules for scheduling the Pod of this application. | {} | | +| `tolerations` | Taints to be tolerated by Pods of this application. | [] | | + + +Check the TVM CR configuration by running following command: + +``` +kubectl get triliovaultmanagers.triliovault.trilio.io triliovault-manager -o yaml +``` + +Once the operator pod is in running state, the TVK pods getting spawned. Confirm the [TVK pods are up](#Check-TVK-Install). + +#### Note: + +If preflight check is enabled and helm install fails, check pre-install helm hook pod logs for any failure in preflight check. Do the following steps: + +First, run this command: +``` +kubectl get pods -n +``` + +The pod name should start with `-preflight-job-preinstall-hook`. Check the logs of the pod by the following command: +``` +kubectl logs -f -n +``` + +#### The failed preflight job is not cleaned up automatically right after failure. If the user cluster version is 1.21 and above, the job will be cleaned up after 1 hour so user should collect any failure logs within 1 hr of job failure. For cluster version below 1.21, user has to clean up failed preflight job manually. + +To delete the job manually, run the following command: +``` +kubectl delete job -f -n +``` + +where job name should also start with `-preflight-job-preinstall-hook` + +Also, due to a bug at helm side where auto deletion of resources upon failure doesn't work, user needs to clean the following resources left behind to be able to run preflight again, until the bug is fixed from their side, after which this step will be handled automatically. Run the following command to clean up the temporary resources: + +1. Cleanup Service Account: + ``` + kubectl delete sa -preflight-service-account -n + ``` +2. Cleanup Cluster Role Binding: + ``` + kubectl delete clusterrolebinding --preflight-rolebinding + ``` +3. Cleanup Cluster Role: + ``` + kubectl delete clusterrole --preflight-role + ``` + +## Manual Installation + +To install the operator on local setup just run the latest helm charts inside this repo + +```shell script +helm repo add trilio-vault-operator https://charts.k8strilio.net/trilio-stable/k8s-triliovault-operator +helm install tvm trilio-vault-operator/k8s-triliovault-operator +``` + +Now, create a TrilioVaultManager CR to install the TrilioVault for Kubernetes. You can provide the custom configurations for the TVK resources as follows: + +``` +apiVersion: triliovault.trilio.io/v1 +kind: TrilioVaultManager +metadata: + labels: + triliovault: k8s + name: tvk +spec: + trilioVaultAppVersion: latest + applicationScope: Cluster + # User can configure tvk instance name + tvkInstanceName: tvk-instance + # User can configure the ingress hosts, annotations and TLS secret through the ingressConfig section + ingressConfig: + host: "trilio.co.in" + tlsSecretName: "secret-name" + # TVK components configuration, currently supports control-plane, web, exporter, web-backend, ingress-controller, admission-webhook. + # User can configure resources for all componentes and can configure service type and host for the ingress-controller + componentConfiguration: + web-backend: + resources: + requests: + memory: "400Mi" + cpu: "200m" + limits: + memory: "2584Mi" + cpu: "1000m" + ingress-controller: + enabled: true + service: + type: LoadBalancer +``` + +### Apply the Custom Resource + +Apply `TVM.yaml`: + +```shell +kubectl create -f TVM.yaml +``` + +### Check TVK Install + +Check that the pods were created: + +``` +kubectl get pods +``` + +``` +NAME READY STATUS RESTARTS AGE +k8s-triliovault-admission-webhook-6ff5f98c8-qwmfc 1/1 Running 0 81s +k8s-triliovault-backend-6f66b6b8d5-gxtmz 1/1 Running 0 81s +k8s-triliovault-control-plane-6c464c5d78-ftk6g 1/1 Running 0 81s +k8s-triliovault-exporter-59566f97dd-gs4xc 1/1 Running 0 81s +k8s-triliovault-ingress-nginx-controller-867c764cd5-qhpx6 1/1 Running 0 18s +k8s-triliovault-web-967c8475-m7pc6 1/1 Running 0 81s +tvm-k8s-triliovault-operator-66bd7d86d5-dvhzb 1/1 Running 0 6m48s +``` + +Check that ingress controller service is of type LoadBalancer: +``` +NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE +k8s-triliovault-admission-webhook ClusterIP 10.7.243.24 443/TCP 129m +k8s-triliovault-ingress-nginx-controller LoadBalancer 10.7.246.193 35.203.155.148 80:30362/TCP,443:32327/TCP 129m +k8s-triliovault-ingress-nginx-controller-admission ClusterIP 10.7.250.31 443/TCP 129m +k8s-triliovault-web ClusterIP 10.7.254.41 80/TCP 129m +k8s-triliovault-web-backend ClusterIP 10.7.252.146 80/TCP 129m +tvm-k8s-triliovault-operator-webhook-service ClusterIP 10.7.248.163 443/TCP 130m 123m +``` + +Check that ingress resources has the host defined by the user: +``` +NAME CLASS HOSTS ADDRESS PORTS AGE +k8s-triliovault k8s-triliovault-default-nginx * 35.203.155.148 80 129m +``` + +You can access the TVK UI by hitting this address in your browser: https://35.203.155.148 + +## Delete + +```shell +kubectl delete -f TVM.yaml +``` + +## Uninstall + +To uninstall/delete the operator helm chart : + +```bash +helm uninstall tvm +``` + +## TrilioVaultManager compatibility + +We maintain the version parity between the TrilioVaultManager(upstream operator) and TrilioVault for Kubernetes. Whenever +user wants to upgrade to the new version, should use the same version for upstream operator and Triliovault for Kubernetes. diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/Chart.yaml new file mode 100644 index 000000000..4df538147 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/Chart.yaml @@ -0,0 +1,21 @@ +apiVersion: v2 +appVersion: 0.1.0 +dependencies: +- name: visualization + repository: file://charts/visualization + version: ^0.1.0 +- name: logging + repository: file://charts/logging + version: ^0.1.0 +- name: monitoring + repository: file://charts/monitoring + version: ^0.1.0 +description: Observability Stack is designed to manage the K8s-TrilioVault Application's + Logging, Monitoring and Visualization. +icon: https://www.trilio.io/wp-content/uploads/2021/01/Trilio-2020-logo-RGB-gray-green.png +kubeVersion: '>=1.19.0-0' +maintainers: +- email: support@trilio.io + name: Trilio +name: observability +version: 0.1.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/Chart.yaml new file mode 100644 index 000000000..411d5b0fd --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/Chart.yaml @@ -0,0 +1,18 @@ +apiVersion: v2 +appVersion: 0.1.0 +dependencies: +- condition: loki.enabled + name: loki + repository: https://grafana.github.io/helm-charts + version: ^2.15.2 +- condition: promtail.enabled + name: promtail + repository: https://grafana.github.io/helm-charts + version: ^6.7.4 +description: Logging Stack designed to manage the K8s-TrilioVault Application's Logs. +icon: https://www.trilio.io/wp-content/uploads/2021/01/Trilio-2020-logo-RGB-gray-green.png +maintainers: +- email: support@trilio.io + name: Trilio +name: logging +version: 0.1.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/Chart.yaml new file mode 100644 index 000000000..c10bab260 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/Chart.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +appVersion: v2.6.1 +description: 'Loki: like Prometheus, but for logs.' +home: https://grafana.com/loki +icon: https://raw.githubusercontent.com/grafana/loki/master/docs/sources/logo.png +kubeVersion: ^1.10.0-0 +maintainers: +- email: support@trilio.io + name: Trilio +name: loki +sources: +- https://github.com/grafana/loki +version: 2.16.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/NOTES.txt new file mode 100644 index 000000000..abe023a70 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/NOTES.txt @@ -0,0 +1,3 @@ +Verify the application is working by running these commands: + kubectl --namespace {{ .Release.Namespace }} port-forward service/{{ include "loki.fullname" . }} {{ .Values.service.port }} + curl http://127.0.0.1:{{ .Values.service.port }}/api/prom/label diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/_helpers.tpl new file mode 100644 index 000000000..1ff9b632a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/_helpers.tpl @@ -0,0 +1,99 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "loki.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 "loki.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 "loki.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account +*/}} +{{- define "loki.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "loki.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the app name of loki clients. Defaults to the same logic as "loki.fullname", and default client expects "promtail". +*/}} +{{- define "client.name" -}} +{{- if .Values.client.name -}} +{{- .Values.client.name -}} +{{- else if .Values.client.fullnameOverride -}} +{{- .Values.client.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default "promtail" .Values.client.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Generate a right Ingress apiVersion +*/}} +{{- define "ingress.apiVersion" -}} +{{- if semverCompare ">=1.20-0" .Capabilities.KubeVersion.GitVersion -}} +networking.k8s.io/v1 +{{- else if semverCompare ">=1.14-0" .Capabilities.KubeVersion.GitVersion -}} +networking.k8s.io/v1beta1 +{{- else -}} +extensions/v1 +{{- end }} +{{- end -}} + +{{/* +Handle backwards compatible api versions for: + - podDisruptionBudget (policy/v1beta1) + - podSecurityPolicy (policy/v1beta1) +*/}} +{{- define "loki.podDisruptionBudget.apiVersion" -}} +{{ if $.Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudgets" -}} +{{- print "policy/v1" -}} +{{- else -}} +{{- print "policy/v1beta1" -}} +{{- end -}} +{{- end -}} + + +{{/* +Common labels +*/}} +{{- define "loki.labels" -}} +app: {{ template "loki.name" . }} +chart: {{ template "loki.chart" . }} +release: {{ .Release.Name }} +heritage: {{ .Release.Service }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/configmap-alert.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/configmap-alert.yaml new file mode 100644 index 000000000..45d72d8aa --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/configmap-alert.yaml @@ -0,0 +1,14 @@ +{{- if or (.Values.useExistingAlertingGroup.enabled) (gt (len .Values.alerting_groups) 0) }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "loki.fullname" . }}-alerting-rules + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +data: + {{ template "loki.fullname" . }}-alerting-rules.yaml: |- + groups: + {{- toYaml .Values.alerting_groups | nindent 6 }} +{{- end }} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/ingress.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/ingress.yaml new file mode 100644 index 000000000..b12551437 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/ingress.yaml @@ -0,0 +1,52 @@ +{{- if .Values.ingress.enabled -}} +{{- $fullName := include "loki.fullname" . -}} +{{- $svcPort := .Values.service.port -}} +{{- $apiVersion := include "ingress.apiVersion" . -}} +apiVersion: {{ $apiVersion }} +kind: Ingress +metadata: + name: {{ $fullName }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} + {{- with .Values.ingress.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: +{{- if .Values.ingress.ingressClassName }} + ingressClassName: {{ .Values.ingress.ingressClassName }} +{{- end }} +{{- if .Values.ingress.tls }} + tls: + {{- range .Values.ingress.tls }} + - hosts: + {{- range .hosts }} + - {{ . | quote }} + {{- end }} + secretName: {{ .secretName }} + {{- end }} +{{- end }} + rules: + {{- range .Values.ingress.hosts }} + - host: {{ .host | quote }} + http: + paths: + {{- range .paths }} + - path: {{ . }} + {{- if eq $apiVersion "networking.k8s.io/v1" }} + pathType: Prefix + {{- end }} + backend: + {{- if eq $apiVersion "networking.k8s.io/v1" }} + service: + name: {{ $fullName }} + port: + number: {{ $svcPort }} + {{- else }} + serviceName: {{ $fullName }} + servicePort: {{ $svcPort }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/networkpolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/networkpolicy.yaml new file mode 100644 index 000000000..b9349ba9d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/networkpolicy.yaml @@ -0,0 +1,23 @@ +{{- if .Values.networkPolicy.enabled }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + name: {{ template "loki.fullname" . }} + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} + ingress: + - from: + - podSelector: + matchLabels: + app: {{ template "client.name" . }} + release: {{ .Release.Name }} + - ports: + - port: {{ .Values.service.port }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/pdb.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/pdb.yaml new file mode 100644 index 000000000..68dd619b7 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.podDisruptionBudget -}} +apiVersion: {{ include "loki.podDisruptionBudget.apiVersion" . }} +kind: PodDisruptionBudget +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + app: {{ template "loki.name" . }} +{{ toYaml .Values.podDisruptionBudget | indent 2 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/podsecuritypolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..c30ab49c1 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/podsecuritypolicy.yaml @@ -0,0 +1,40 @@ +{{- if .Values.rbac.pspEnabled }} +{{- if .Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy" }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ template "loki.fullname" . }} + labels: + {{- include "loki.labels" . | nindent 4 }} +spec: + privileged: false + allowPrivilegeEscalation: false + volumes: + - 'configMap' + - 'emptyDir' + - 'persistentVolumeClaim' + - 'secret' + - 'projected' + - 'downwardAPI' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'MustRunAsNonRoot' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: true + requiredDropCapabilities: + - ALL +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/prometheusrule.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/prometheusrule.yaml new file mode 100644 index 000000000..d1ed09be5 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/prometheusrule.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.serviceMonitor.enabled .Values.serviceMonitor.prometheusRule.enabled -}} +apiVersion: monitoring.coreos.com/v1 +kind: PrometheusRule +metadata: + name: {{ template "loki.fullname" . }} +{{- if .Values.serviceMonitor.prometheusRule.namespace }} + namespace: {{ .Values.serviceMonitor.prometheusRule.namespace | quote }} +{{- end }} + labels: + {{- include "loki.labels" . | nindent 4 }} + {{- if .Values.serviceMonitor.prometheusRule.additionalLabels }} + {{- toYaml .Values.serviceMonitor.prometheusRule.additionalLabels | nindent 4 }} + {{- end }} +spec: +{{- if .Values.serviceMonitor.prometheusRule.rules }} + groups: + - name: {{ template "loki.fullname" . }} + rules: {{- toYaml .Values.serviceMonitor.prometheusRule.rules | nindent 4 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/role.yaml new file mode 100644 index 000000000..03b9da608 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/role.yaml @@ -0,0 +1,17 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +{{- if .Values.rbac.pspEnabled }} +rules: +- apiGroups: ['extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: [{{ template "loki.fullname" . }}] +{{- end }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/rolebinding.yaml new file mode 100644 index 000000000..099111de3 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/rolebinding.yaml @@ -0,0 +1,17 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ template "loki.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ template "loki.serviceAccountName" . }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/secret.yaml new file mode 100644 index 000000000..b4bee6a68 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/secret.yaml @@ -0,0 +1,11 @@ +{{- if not .Values.config.existingSecret -}} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +data: + loki.yaml: {{ tpl (toYaml .Values.config) . | b64enc}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-headless.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-headless.yaml new file mode 100644 index 000000000..d97c36a20 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-headless.yaml @@ -0,0 +1,26 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "loki.fullname" . }}-headless + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ template "loki.name" . }} + {{- include "k8s-triliovault-operator.observability" . | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + variant: headless +spec: + clusterIP: None + ports: + - port: {{ .Values.service.port }} + protocol: TCP + name: http-metrics + targetPort: {{ .Values.service.targetPort }} +{{- if .Values.extraPorts }} +{{ toYaml .Values.extraPorts | indent 4}} +{{- end }} + selector: + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-memberlist.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-memberlist.yaml new file mode 100644 index 000000000..27a885785 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service-memberlist.yaml @@ -0,0 +1,21 @@ +{{- if .Values.config.memberlist -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "loki.fullname" . }}-memberlist + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} +spec: + type: ClusterIP + clusterIP: None + publishNotReadyAddresses: true + ports: + - name: http + port: {{ .Values.config.memberlist.bind_port | default 7946 }} + targetPort: memberlist-port + protocol: TCP + selector: + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service.yaml new file mode 100644 index 000000000..1a8877924 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/service.yaml @@ -0,0 +1,43 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} + {{- with .Values.service.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + annotations: + {{- toYaml .Values.service.annotations | nindent 4 }} +spec: +{{- if .Values.service.loadBalancerSourceRanges }} + externalTrafficPolicy: {{ .Values.service.externalTrafficPolicy }} +{{- end }} + type: {{ .Values.service.type }} +{{- if (and (eq .Values.service.type "ClusterIP") (not (empty .Values.service.clusterIP))) }} + clusterIP: {{ .Values.service.clusterIP }} +{{- end }} +{{- if (and (eq .Values.service.type "LoadBalancer") (not (empty .Values.service.loadBalancerIP))) }} + loadBalancerIP: {{ .Values.service.loadBalancerIP }} +{{- end }} +{{- if .Values.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - port: {{ .Values.service.port }} + protocol: TCP + name: http-metrics + targetPort: {{ .Values.service.targetPort }} +{{- if (and (eq .Values.service.type "NodePort") (not (empty .Values.service.nodePort))) }} + nodePort: {{ .Values.service.nodePort }} +{{- end }} +{{- if .Values.extraPorts }} +{{ toYaml .Values.extraPorts | indent 4}} +{{- end }} + selector: + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/serviceaccount.yaml new file mode 100644 index 000000000..6db005d14 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "loki.labels" . | nindent 4 }} + annotations: + {{- toYaml .Values.serviceAccount.annotations | nindent 4 }} + name: {{ template "loki.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +automountServiceAccountToken: {{ .Values.serviceAccount.automountServiceAccountToken }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/servicemonitor.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/servicemonitor.yaml new file mode 100644 index 000000000..dd84a3a45 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/servicemonitor.yaml @@ -0,0 +1,42 @@ +{{- if .Values.serviceMonitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ template "loki.fullname" . }} + labels: + {{- include "loki.labels" . | nindent 4 }} + {{- if .Values.serviceMonitor.additionalLabels }} +{{ toYaml .Values.serviceMonitor.additionalLabels | indent 4 }} + {{- end }} + {{- if .Values.serviceMonitor.annotations }} + annotations: +{{ toYaml .Values.serviceMonitor.annotations | indent 4 }} + {{- end }} +spec: + selector: + matchLabels: + app: {{ template "loki.name" . }} + release: {{ .Release.Name | quote }} + variant: headless + namespaceSelector: + matchNames: + - {{ .Release.Namespace | quote }} + endpoints: + - port: http-metrics + {{- if .Values.serviceMonitor.interval }} + interval: {{ .Values.serviceMonitor.interval }} + {{- end }} + {{- if .Values.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.serviceMonitor.scrapeTimeout }} + {{- end }} + {{- if .Values.serviceMonitor.path }} + path: {{ .Values.serviceMonitor.path }} + {{- end }} + {{- with .Values.serviceMonitor.scheme }} + scheme: {{ . }} + {{- end }} + {{- with .Values.serviceMonitor.tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 6 }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/statefulset.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/statefulset.yaml new file mode 100644 index 000000000..9e1a49ca1 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/templates/statefulset.yaml @@ -0,0 +1,180 @@ +apiVersion: apps/v1 +kind: StatefulSet +metadata: + name: {{ template "loki.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "loki.labels" . | nindent 4 }} + app.kubernetes.io/instance: {{ .Release.Name }} + {{- include "k8s-triliovault-operator.observability" . | nindent 4 }} + annotations: + {{- toYaml .Values.annotations | nindent 4 }} +spec: + podManagementPolicy: {{ .Values.podManagementPolicy }} + replicas: {{ .Values.replicas }} + selector: + matchLabels: + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} + app.kubernetes.io/instance: {{ .Release.Name }} + {{- include "k8s-triliovault-operator.observability" . | nindent 6 }} + serviceName: {{ template "loki.fullname" . }}-headless + updateStrategy: + {{- toYaml .Values.updateStrategy | nindent 4 }} + template: + metadata: + labels: + app: {{ template "loki.name" . }} + name: {{ template "loki.fullname" . }} + release: {{ .Release.Name }} + app.kubernetes.io/instance: {{ .Release.Name }} + {{- include "k8s-triliovault-operator.observability" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + annotations: + {{- if not .Values.config.existingSecret }} + checksum/config: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + serviceAccountName: {{ template "loki.serviceAccountName" . }} + {{- if .Values.priorityClassName }} + priorityClassName: {{ .Values.priorityClassName }} + {{- end }} + securityContext: + {{- toYaml .Values.securityContext | nindent 8 }} + initContainers: + {{- toYaml .Values.initContainers | nindent 8 }} + {{- if .Values.image.pullSecrets }} + imagePullSecrets: + {{- range .Values.image.pullSecrets }} + - name: {{ . }} + {{- end}} + {{- end }} + containers: + - name: {{ .Chart.Name }} + image: "{{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + args: + - "-config.file=/etc/loki/loki.yaml" + {{- range $key, $value := .Values.extraArgs }} + - "-{{ $key }}={{ $value }}" + {{- end }} + volumeMounts: + - name: tmp + mountPath: /tmp + {{- if .Values.extraVolumeMounts }} + {{ toYaml .Values.extraVolumeMounts | nindent 12}} + {{- end }} + - name: config + mountPath: /etc/loki + - name: storage + mountPath: "/data" + subPath: {{ .Values.persistence.subPath }} + {{- if or (.Values.useExistingAlertingGroup.enabled) (gt (len .Values.alerting_groups) 0) }} + - name: rules + mountPath: /rules/fake + {{- end }} + ports: + - name: http-metrics + containerPort: {{ .Values.config.server.http_listen_port | default 3100 }} + protocol: TCP + - name: grpc + containerPort: {{ .Values.config.server.grpc_listen_port | default 9095 }} + protocol: TCP + {{- if .Values.config.memberlist }} + - name: memberlist-port + containerPort: {{ .Values.config.memberlist.bind_port | default 7946 }} + protocol: TCP + {{- end }} + livenessProbe: + {{- toYaml .Values.livenessProbe | nindent 12 }} + readinessProbe: + {{- toYaml .Values.readinessProbe | nindent 12 }} + resources: + {{- toYaml .Values.resources | nindent 12 }} + securityContext: + {{- toYaml .Values.containerSecurityContext | nindent 12 }} + env: + {{- if .Values.env }} + {{- toYaml .Values.env | nindent 12 }} + {{- end }} + {{- if .Values.tracing.jaegerAgentHost }} + - name: JAEGER_AGENT_HOST + value: "{{ .Values.tracing.jaegerAgentHost }}" + {{- end }} + {{- with .Values.extraEnvFrom }} + envFrom: + {{- toYaml . | nindent 12 }} + {{- end }} +{{- if .Values.extraContainers }} +{{ toYaml .Values.extraContainers | indent 8}} +{{- end }} + nodeSelector: + {{- toYaml .Values.nodeSelector | nindent 8 }} + affinity: + {{- toYaml .Values.affinity | nindent 8 }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- if .Values.topologySpreadConstraints.enabled }} + topologySpreadConstraints: + - maxSkew: {{ .Values.topologySpreadConstraints.maxSkew | default 1 }} + topologyKey: {{ .Values.topologySpreadConstraints.topologyKey | default "topology.kubernetes.io/zone" }} + whenUnsatisfiable: {{ .Values.topologySpreadConstraints.whenUnsatisfiable | default "ScheduleAnyway" }} + matchLabels: + app: {{ template "loki.name" . }} + release: {{ .Release.Name }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} + volumes: + - name: tmp + emptyDir: {} + {{- if or (.Values.useExistingAlertingGroup.enabled) (gt (len .Values.alerting_groups) 0) }} + - name: rules + configMap: + {{- if .Values.useExistingAlertingGroup.enabled }} + name: {{ .Values.useExistingAlertingGroup.configmapName }} + {{- else }} + name: {{ template "loki.fullname" . }}-alerting-rules + {{- end }} + {{- end }} + - name: config + secret: + {{- if .Values.config.existingSecret }} + secretName: {{ .Values.config.existingSecret }} + {{- else }} + secretName: {{ template "loki.fullname" . }} + {{- end }} +{{- if .Values.extraVolumes }} +{{ toYaml .Values.extraVolumes | indent 8}} +{{- end }} + {{- if not .Values.persistence.enabled }} + - name: storage + emptyDir: {} + {{- else if .Values.persistence.existingClaim }} + - name: storage + persistentVolumeClaim: + claimName: {{ .Values.persistence.existingClaim }} + {{- else }} + volumeClaimTemplates: + - metadata: + name: storage + labels: + {{- toYaml .Values.persistence.labels | nindent 8 }} + annotations: + {{- toYaml .Values.persistence.annotations | nindent 8 }} + spec: + accessModes: + {{- toYaml .Values.persistence.accessModes | nindent 8 }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + storageClassName: {{ .Values.persistence.storageClassName }} + {{- if .Values.persistence.selector }} + selector: + {{- toYaml .Values.persistence.selector | nindent 8 }} + {{- end }} + {{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/values.yaml new file mode 100644 index 000000000..731c4d906 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/loki/values.yaml @@ -0,0 +1,346 @@ +image: + registry: docker.io + repository: grafana/loki + tag: 2.6.1 + pullPolicy: IfNotPresent + + ## Optionally specify an array of imagePullSecrets. + ## Secrets must be manually created in the namespace. + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + ## + # pullSecrets: + # - myRegistryKeySecretName + +ingress: + enabled: false + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + hosts: + - host: chart-example.local + paths: [] + tls: [] + # - secretName: chart-example-tls + # hosts: + # - chart-example.local + +## Affinity for pod assignment +## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity +affinity: {} +# podAntiAffinity: +# requiredDuringSchedulingIgnoredDuringExecution: +# - labelSelector: +# matchExpressions: +# - key: app +# operator: In +# values: +# - loki +# topologyKey: "kubernetes.io/hostname" + +## StatefulSet annotations +annotations: {} + +# enable tracing for debug, need install jaeger and specify right jaeger_agent_host +tracing: + jaegerAgentHost: + +config: + # existingSecret: + auth_enabled: false + + memberlist: + join_members: + # the value must be defined as string to be evaluated when secret manifest is being generating + - '{{ include "loki.fullname" . }}-memberlist' + + ingester: + chunk_idle_period: 3m + chunk_block_size: 262144 + chunk_retain_period: 1m + max_transfer_retries: 0 + wal: + dir: /data/loki/wal + lifecycler: + ring: + replication_factor: 1 + + ## Different ring configs can be used. E.g. Consul + # ring: + # store: consul + # replication_factor: 1 + # consul: + # host: "consul:8500" + # prefix: "" + # http_client_timeout: "20s" + # consistent_reads: true + limits_config: + enforce_metric_name: false + reject_old_samples: true + reject_old_samples_max_age: 168h + max_entries_limit_per_query: 5000 + max_query_length: 0h + schema_config: + configs: + - from: 2020-10-24 + store: boltdb-shipper + object_store: filesystem + schema: v11 + index: + prefix: index_ + period: 24h + server: + http_listen_port: 3100 + grpc_listen_port: 9095 + storage_config: + boltdb_shipper: + active_index_directory: /data/loki/boltdb-shipper-active + cache_location: /data/loki/boltdb-shipper-cache + cache_ttl: 24h # Can be increased for faster performance over longer query periods, uses more disk space + shared_store: filesystem + filesystem: + directory: /data/loki/chunks + chunk_store_config: + max_look_back_period: 0s + table_manager: + retention_deletes_enabled: true + retention_period: 168h + compactor: + working_directory: /data/loki/boltdb-shipper-compactor + shared_store: filesystem +# Needed for Alerting: https://grafana.com/docs/loki/latest/rules/ +# This is just a simple example, for more details: https://grafana.com/docs/loki/latest/configuration/#ruler_config +# ruler: +# storage: +# type: local +# local: +# directory: /rules +# rule_path: /tmp/scratch +# alertmanager_url: http://alertmanager.svc.namespace:9093 +# ring: +# kvstore: +# store: inmemory +# enable_api: true + +## Additional Loki container arguments, e.g. log level (debug, info, warn, error) +extraArgs: {} + # log.level: debug + +extraEnvFrom: [] + +livenessProbe: + httpGet: + path: /ready + port: http-metrics + initialDelaySeconds: 45 + +## ref: https://kubernetes.io/docs/concepts/services-networking/network-policies/ +networkPolicy: + enabled: false + +## The app name of loki clients +client: {} + # name: + +## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ +nodeSelector: {} + +## ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/ +## If you set enabled as "True", you need : +## - create a pv which above 10Gi and has same namespace with loki +## - keep storageClassName same with below setting +persistence: + enabled: false + accessModes: + - ReadWriteOnce + size: 10Gi + labels: {} + annotations: {} + # selector: + # matchLabels: + # app.kubernetes.io/name: loki + # subPath: "" + # existingClaim: + # storageClassName: + +## Pod Labels +podLabels: {} + +## Pod Annotations +podAnnotations: + prometheus.io/scrape: "true" + prometheus.io/port: "http-metrics" + +podManagementPolicy: OrderedReady + +## Assign a PriorityClassName to pods if set +# priorityClassName: + +rbac: + create: true + pspEnabled: false + +readinessProbe: + httpGet: + path: /ready + port: http-metrics + initialDelaySeconds: 45 + +replicas: 1 + +resources: + limits: + cpu: 400m + memory: 512Mi + requests: + cpu: 100m + memory: 128Mi + +securityContext: + fsGroup: 10001 + runAsGroup: 10001 + runAsNonRoot: true + runAsUser: 10001 + +containerSecurityContext: + readOnlyRootFilesystem: true + +service: + type: ClusterIP + nodePort: + port: 3100 + annotations: {} + labels: {} + targetPort: http-metrics + +serviceAccount: + create: true + name: + annotations: {} + automountServiceAccountToken: true + +terminationGracePeriodSeconds: 4800 + +## Tolerations for pod assignment +## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +tolerations: [] + +## Topology spread constraint for multi-zone clusters +## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/ +topologySpreadConstraints: + enabled: false + +# The values to set in the PodDisruptionBudget spec +# If not set then a PodDisruptionBudget will not be created +podDisruptionBudget: {} +# minAvailable: 1 +# maxUnavailable: 1 + +updateStrategy: + type: RollingUpdate + +serviceMonitor: + enabled: false + interval: "" + additionalLabels: {} + annotations: {} + # scrapeTimeout: 10s + # path: /metrics + scheme: null + tlsConfig: {} + prometheusRule: + enabled: false + additionalLabels: {} + # namespace: + rules: [] + # Some examples from https://awesome-prometheus-alerts.grep.to/rules.html#loki + # - alert: LokiProcessTooManyRestarts + # expr: changes(process_start_time_seconds{job=~"loki"}[15m]) > 2 + # for: 0m + # labels: + # severity: warning + # annotations: + # summary: Loki process too many restarts (instance {{ $labels.instance }}) + # description: "A loki process had too many restarts (target {{ $labels.instance }})\n VALUE = {{ $value }}\n LABELS = {{ $labels }}" + # - alert: LokiRequestErrors + # expr: 100 * sum(rate(loki_request_duration_seconds_count{status_code=~"5.."}[1m])) by (namespace, job, route) / sum(rate(loki_request_duration_seconds_count[1m])) by (namespace, job, route) > 10 + # for: 15m + # labels: + # severity: critical + # annotations: + # summary: Loki request errors (instance {{ $labels.instance }}) + # description: "The {{ $labels.job }} and {{ $labels.route }} are experiencing errors\n VALUE = {{ $value }}\n LABELS = {{ $labels }}" + # - alert: LokiRequestPanic + # expr: sum(increase(loki_panic_total[10m])) by (namespace, job) > 0 + # for: 5m + # labels: + # severity: critical + # annotations: + # summary: Loki request panic (instance {{ $labels.instance }}) + # description: "The {{ $labels.job }} is experiencing {{ printf \"%.2f\" $value }}% increase of panics\n VALUE = {{ $value }}\n LABELS = {{ $labels }}" + # - alert: LokiRequestLatency + # expr: (histogram_quantile(0.99, sum(rate(loki_request_duration_seconds_bucket{route!~"(?i).*tail.*"}[5m])) by (le))) > 1 + # for: 5m + # labels: + # severity: critical + # annotations: + # summary: Loki request latency (instance {{ $labels.instance }}) + # description: "The {{ $labels.job }} {{ $labels.route }} is experiencing {{ printf \"%.2f\" $value }}s 99th percentile latency\n VALUE = {{ $value }}\n LABELS = {{ $labels }}" + + +initContainers: [] +## Init containers to be added to the loki pod. +# - name: my-init-container +# image: busybox:latest +# command: ['sh', '-c', 'echo hello'] + +extraContainers: [] +## Additional containers to be added to the loki pod. +# - name: reverse-proxy +# image: angelbarrera92/basic-auth-reverse-proxy:dev +# args: +# - "serve" +# - "--upstream=http://localhost:3100" +# - "--auth-config=/etc/reverse-proxy-conf/authn.yaml" +# ports: +# - name: http +# containerPort: 11811 +# protocol: TCP +# volumeMounts: +# - name: reverse-proxy-auth-config +# mountPath: /etc/reverse-proxy-conf + + +extraVolumes: [] +## Additional volumes to the loki pod. +# - name: reverse-proxy-auth-config +# secret: +# secretName: reverse-proxy-auth-config + +## Extra volume mounts that will be added to the loki container +extraVolumeMounts: [] + +extraPorts: [] +## Additional ports to the loki services. Useful to expose extra container ports. +# - port: 11811 +# protocol: TCP +# name: http +# targetPort: http + +# Extra env variables to pass to the loki container +env: [] + +# Specify Loki Alerting rules based on this documentation: https://grafana.com/docs/loki/latest/rules/ +# When specified, you also need to add a ruler config section above. An example is shown in the rules docs. +alerting_groups: [] +# - name: example +# rules: +# - alert: HighThroughputLogStreams +# expr: sum by(container) (rate({job=~"loki-dev/.*"}[1m])) > 1000 +# for: 2m + +useExistingAlertingGroup: + enabled: false + configmapName: "" diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/Chart.yaml new file mode 100644 index 000000000..d69c5a044 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/Chart.yaml @@ -0,0 +1,16 @@ +apiVersion: v2 +appVersion: 2.7.2 +description: Promtail is an agent which ships the contents of local logs to a Loki + instance +home: https://grafana.com/loki +icon: https://raw.githubusercontent.com/grafana/loki/master/docs/sources/logo.png +maintainers: +- email: support@trilio.io + name: Trilio +name: promtail +sources: +- https://github.com/grafana/loki +- https://grafana.com/oss/loki/ +- https://grafana.com/docs/loki/latest/ +type: application +version: 6.8.2 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/NOTES.txt new file mode 100644 index 000000000..01bf66b7f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/NOTES.txt @@ -0,0 +1,15 @@ +*********************************************************************** + Welcome to Grafana Promtail + Chart version: {{ .Chart.Version }} + Promtail version: {{ .Values.image.tag | default .Chart.AppVersion }} +*********************************************************************** + +Verify the application is working by running these commands: + +{{- if .Values.daemonset.enabled }} +* kubectl --namespace {{ .Release.Namespace }} port-forward daemonset/{{ include "promtail.fullname" . }} {{ .Values.config.serverPort }} +{{- end }} +{{- if .Values.deployment.enabled }} +* kubectl --namespace {{ .Release.Namespace }} port-forward deployment/{{ include "promtail.fullname" . }} {{ .Values.config.serverPort }} +{{- end }} +* curl http://127.0.0.1:{{ .Values.config.serverPort }}/metrics diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_helpers.tpl new file mode 100644 index 000000000..59053c253 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_helpers.tpl @@ -0,0 +1,102 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "promtail.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 "promtail.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 "promtail.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "promtail.labels" -}} +helm.sh/chart: {{ include "promtail.chart" . }} +{{ include "promtail.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "promtail.selectorLabels" -}} +app.kubernetes.io/name: {{ include "promtail.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{ include "k8s-triliovault-operator.observability" .}} +{{- end }} + +{{/* +Create the name of the namespace +*/}} +{{- define "promtail.namespaceName" -}} +{{- default .Release.Namespace .Values.namespace }} +{{- end }} + +{{/* +Create the name of the service account +*/}} +{{- define "promtail.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "promtail.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +The service name to connect to Loki. Defaults to the same logic as "loki.fullname" +*/}} +{{- define "loki.serviceName" -}} +{{- if .Values.loki.serviceName }} +{{- .Values.loki.serviceName }} +{{- else if .Values.loki.fullnameOverride }} +{{- .Values.loki.fullnameOverride | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- $name := default "loki" .Values.loki.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Configure enableServiceLinks in pod +*/}} +{{- define "promtail.enableServiceLinks" -}} +{{- if semverCompare ">=1.13-0" .Capabilities.KubeVersion.GitVersion }} +{{- if or (.Values.enableServiceLinks) (eq (.Values.enableServiceLinks | toString) "") }} +{{- printf "enableServiceLinks: true" }} +{{- else }} +{{- printf "enableServiceLinks: false" }} +{{- end }} +{{- end }} +{{- end }} + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_pod.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_pod.tpl new file mode 100644 index 000000000..d5a14b411 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/_pod.tpl @@ -0,0 +1,118 @@ +{{/* +Pod template used in Daemonset and Deployment +*/}} +{{- define "promtail.podTemplate" -}} +metadata: + labels: + {{- include "promtail.selectorLabels" . | nindent 4 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + annotations: + checksum/config: {{ include (print .Template.BasePath "/secret.yaml") . | sha256sum }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + serviceAccountName: {{ include "promtail.serviceAccountName" . }} + {{- include "promtail.enableServiceLinks" . | nindent 2 }} + {{- with .Values.priorityClassName }} + priorityClassName: {{ . }} + {{- end }} + {{- with .Values.initContainer }} + initContainers: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 4 }} + {{- end }} + securityContext: + {{- toYaml .Values.podSecurityContext | nindent 4 }} + containers: + - name: promtail + image: "{{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }}" + imagePullPolicy: {{ .Values.image.pullPolicy }} + args: + - "-config.file=/etc/promtail/promtail.yaml" + {{- with .Values.extraArgs }} + {{- toYaml . | nindent 8 }} + {{- end }} + volumeMounts: + - name: config + mountPath: /etc/promtail + {{- with .Values.defaultVolumeMounts }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.extraVolumeMounts }} + {{- toYaml . | nindent 8 }} + {{- end }} + env: + - name: HOSTNAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + {{- with .Values.extraEnv }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.extraEnvFrom }} + envFrom: + {{- toYaml . | nindent 8 }} + {{- end }} + ports: + - name: http-metrics + containerPort: {{ .Values.config.serverPort }} + protocol: TCP + {{- range $key, $values := .Values.extraPorts }} + - name: {{ .name | default $key }} + containerPort: {{ $values.containerPort }} + protocol: {{ $values.protocol | default "TCP" }} + {{- end }} + securityContext: + {{- toYaml .Values.containerSecurityContext | nindent 8 }} + {{- with .Values.livenessProbe }} + livenessProbe: + {{- tpl (toYaml .) $ | nindent 8 }} + {{- end }} + {{- with .Values.readinessProbe }} + readinessProbe: + {{- tpl (toYaml .) $ | nindent 8 }} + {{- end }} + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.extraContainers }} + {{- range $name, $values := .Values.extraContainers }} + - name: {{ $name }} + {{ toYaml $values | nindent 6 }} + {{- end }} + {{- end }} + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 4 }} + {{- end }} + volumes: + - name: config + {{- if .Values.configmap.enabled }} + configMap: + name: {{ include "promtail.fullname" . }} + {{- else }} + secret: + secretName: {{ include "promtail.fullname" . }} + {{- end }} + {{- with .Values.defaultVolumes }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.extraVolumes }} + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrole.yaml new file mode 100644 index 000000000..4702e60d0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrole.yaml @@ -0,0 +1,21 @@ +{{- if .Values.rbac.create }} +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "promtail.fullname" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - nodes + - nodes/proxy + - services + - endpoints + - pods + verbs: + - get + - watch + - list +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..e92bf9a6d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if .Values.rbac.create }} +kind: ClusterRoleBinding +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: {{ include "promtail.fullname" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +subjects: + - kind: ServiceAccount + name: {{ include "promtail.serviceAccountName" . }} + namespace: {{ include "promtail.namespaceName" . }} +roleRef: + kind: ClusterRole + name: {{ include "promtail.fullname" . }} + apiGroup: rbac.authorization.k8s.io +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/configmap.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/configmap.yaml new file mode 100644 index 000000000..0785b1a60 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/configmap.yaml @@ -0,0 +1,12 @@ +{{- if .Values.configmap.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "promtail.fullname" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +data: + promtail.yaml: | + {{- tpl .Values.config.file . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/daemonset.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/daemonset.yaml new file mode 100644 index 000000000..85a8aa031 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/daemonset.yaml @@ -0,0 +1,21 @@ +{{- if .Values.daemonset.enabled }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "promtail.fullname" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + updateStrategy: + {{- toYaml .Values.updateStrategy | nindent 4 }} + template: + {{- include "promtail.podTemplate" . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/deployment.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/deployment.yaml new file mode 100644 index 000000000..26e7381a3 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/deployment.yaml @@ -0,0 +1,22 @@ +{{- if .Values.deployment.enabled }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "promtail.fullname" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} + {{- with .Values.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- if not .Values.deployment.autoscaling.enabled }} + replicas: {{ .Values.deployment.replicaCount }} + {{- end }} + selector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + template: + {{- include "promtail.podTemplate" . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/extra-manifests.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/extra-manifests.yaml new file mode 100644 index 000000000..a9bb3b6ba --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraObjects }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/hpa.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/hpa.yaml new file mode 100644 index 000000000..8a205fde9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/hpa.yaml @@ -0,0 +1,31 @@ +{{- if and .Values.deployment.enabled .Values.deployment.autoscaling.enabled }} +apiVersion: autoscaling/v2beta1 +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "promtail.fullname" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "promtail.fullname" . }} + {{- with .Values.deployment.autoscaling }} + minReplicas: {{ .minReplicas }} + maxReplicas: {{ .maxReplicas }} + metrics: + {{- with .targetCPUUtilizationPercentage }} + - type: Resource + resource: + name: cpu + targetAverageUtilization: {{ . }} + {{- end }} + {{- with .targetMemoryUtilizationPercentage }} + - type: Resource + resource: + name: memory + targetAverageUtilization: {{ . }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/networkpolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/networkpolicy.yaml new file mode 100644 index 000000000..9467df5de --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/networkpolicy.yaml @@ -0,0 +1,123 @@ +{{- if .Values.networkPolicy.enabled }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "promtail.name" . }}-namespace-only + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + podSelector: {} + policyTypes: + - Ingress + - Egress + egress: + - to: + - podSelector: {} + ingress: + - from: + - podSelector: {} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "promtail.name" . }}-egress-dns + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + policyTypes: + - Egress + egress: + - ports: + - port: 53 + protocol: UDP + to: + - namespaceSelector: {} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "promtail.name" . }}-egress-k8s-api + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + policyTypes: + - Egress + egress: + - ports: + - port: {{ .Values.networkPolicy.k8sApi.port }} + protocol: TCP + {{- if len .Values.networkPolicy.k8sApi.cidrs }} + to: + {{- range $cidr := .Values.networkPolicy.k8sApi.cidrs }} + - ipBlock: + cidr: {{ $cidr }} + {{- end }} + {{- end }} + +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "promtail.name" . }}-ingress-metrics + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + policyTypes: + - Ingress + ingress: + - ports: + - port: http-metrics + protocol: TCP + {{- if len .Values.networkPolicy.metrics.cidrs }} + from: + {{- range $cidr := .Values.networkPolicy.metrics.cidrs }} + - ipBlock: + cidr: {{ $cidr }} + {{- end }} + {{- if .Values.networkPolicy.metrics.namespaceSelector }} + - namespaceSelector: + {{- toYaml .Values.networkPolicy.metrics.namespaceSelector | nindent 12 }} + {{- if .Values.networkPolicy.metrics.podSelector }} + podSelector: + {{- toYaml .Values.networkPolicy.metrics.podSelector | nindent 12 }} + {{- end }} + {{- end }} + {{- end }} + +{{- if .Values.extraPorts }} +--- +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "promtail.name" . }}-egress-extra-ports + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "promtail.selectorLabels" . | nindent 6 }} + policyTypes: + - Egress + egress: + - ports: + {{- range $extraPortConfig := .Values.extraPorts }} + - port: {{ $extraPortConfig.containerPort }} + protocol: {{ $extraPortConfig.protocol }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/podsecuritypolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..a22938826 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/podsecuritypolicy.yaml @@ -0,0 +1,10 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.rbac.create .Values.rbac.pspEnabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "promtail.fullname" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + {{- toYaml .Values.podSecurityPolicy | nindent 2 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/prometheus-rules.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/prometheus-rules.yaml new file mode 100644 index 000000000..6ec3f26a9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/prometheus-rules.yaml @@ -0,0 +1,21 @@ +{{- if and .Values.serviceMonitor.enabled .Values.serviceMonitor.prometheusRule.enabled -}} +apiVersion: monitoring.coreos.com/v1 +kind: PrometheusRule +metadata: + name: {{ include "promtail.fullname" . }} + {{- with .Values.serviceMonitor.prometheusRule.namespace }} + namespace: {{ . | quote }} + {{- end }} + labels: + {{- include "promtail.labels" . | nindent 4 }} + {{- with .Values.serviceMonitor.prometheusRule.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: +{{- if .Values.serviceMonitor.prometheusRule.rules }} + groups: + - name: {{ template "promtail.fullname" . }} + rules: + {{- toYaml .Values.serviceMonitor.prometheusRule.rules | nindent 4 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/role.yaml new file mode 100644 index 000000000..a193b3f5b --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/role.yaml @@ -0,0 +1,18 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.rbac.create .Values.rbac.pspEnabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "promtail.fullname" . }}-psp + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +rules: + - apiGroups: + - policy + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ include "promtail.fullname" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/rolebinding.yaml new file mode 100644 index 000000000..0527fdc55 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/rolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy") .Values.rbac.create .Values.rbac.pspEnabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "promtail.fullname" . }}-psp + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "promtail.fullname" . }}-psp +subjects: + - kind: ServiceAccount + name: {{ include "promtail.serviceAccountName" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/secret.yaml new file mode 100644 index 000000000..f5d61ace3 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/secret.yaml @@ -0,0 +1,19 @@ +{{- if not .Values.configmap.enabled }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "promtail.fullname" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} + {{- with .Values.secret.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.secret.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +stringData: + promtail.yaml: | + {{- tpl .Values.config.file . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-extra.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-extra.yaml new file mode 100644 index 000000000..7257e6894 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-extra.yaml @@ -0,0 +1,52 @@ +{{- range $key, $values := .Values.extraPorts }} +--- +apiVersion: v1 +kind: Service +metadata: + name: {{ include "promtail.fullname" $ }}-{{ $key | lower }} + namespace: {{ include "promtail.namespaceName" $ }} + labels: + {{- include "promtail.labels" $ | nindent 4 }} + {{- with .labels }} + {{- toYaml $ | nindent 4 }} + {{- end }} + {{- with .annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- with $values.service }} + type: {{ .type | default "ClusterIP" }} + {{- with .clusterIP }} + clusterIP: {{ . }} + {{- end }} + {{- with .loadBalancerIP }} + loadBalancerIP: {{ . }} + {{- end }} + {{- with .loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .externalIPs }} + externalIPs: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .externalTrafficPolicy }} + externalTrafficPolicy: {{ . }} + {{- end }} + {{- end }} + ports: + - name: {{ .name | default $key }} + targetPort: {{ .name | default $key }} + protocol: {{ $values.protocol | default "TCP" }} + {{- if $values.service }} + port: {{ $values.service.port | default $values.containerPort }} + {{- if $values.service.nodePort }} + nodePort: {{ $values.service.nodePort }} + {{- end }} + {{- else }} + port: {{ $values.containerPort }} + {{- end }} + selector: + {{- include "promtail.selectorLabels" $ | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-metrics.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-metrics.yaml new file mode 100644 index 000000000..4948ceecf --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/service-metrics.yaml @@ -0,0 +1,18 @@ +{{- if .Values.serviceMonitor.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ include "promtail.fullname" . }}-metrics + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} +spec: + clusterIP: None + ports: + - name: http-metrics + port: {{ .Values.config.serverPort }} + targetPort: http-metrics + protocol: TCP + selector: + {{- include "promtail.selectorLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/serviceaccount.yaml new file mode 100644 index 000000000..658c2012f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/serviceaccount.yaml @@ -0,0 +1,17 @@ +{{- if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "promtail.serviceAccountName" . }} + namespace: {{ include "promtail.namespaceName" . }} + labels: + {{- include "promtail.labels" . | nindent 4 }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- with .Values.serviceAccount.imagePullSecrets }} +imagePullSecrets: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/servicemonitor.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/servicemonitor.yaml new file mode 100644 index 000000000..f43964931 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/templates/servicemonitor.yaml @@ -0,0 +1,58 @@ +{{- with .Values.serviceMonitor }} +{{- if .enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "promtail.fullname" $ }} + {{- with .namespace }} + namespace: {{ . }} + {{- end }} + {{- with .annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} + labels: + {{- include "promtail.labels" $ | nindent 4 }} + {{- with .labels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- with .namespaceSelector }} + namespaceSelector: + {{- toYaml . | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- include "promtail.selectorLabels" $ | nindent 6 }} + endpoints: + - port: http-metrics + {{- with $.Values.httpPathPrefix }} + path: {{ printf "%s/metrics" . }} + {{- end }} + {{- with .interval }} + interval: {{ . }} + {{- end }} + {{- with .scrapeTimeout }} + scrapeTimeout: {{ . }} + {{- end }} + {{- with .relabelings }} + relabelings: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .metricRelabelings }} + metricRelabelings: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .scheme }} + scheme: {{ . }} + {{- end }} + {{- with .tlsConfig }} + tlsConfig: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .targetLabels }} + targetLabels: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/values.yaml new file mode 100644 index 000000000..58d7752ea --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/charts/promtail/values.yaml @@ -0,0 +1,534 @@ +# -- Overrides the chart's name +nameOverride: null + +# -- Overrides the chart's computed fullname +fullnameOverride: null + +daemonset: + # -- Deploys Promtail as a DaemonSet + enabled: true + +deployment: + # -- Deploys Promtail as a Deployment + enabled: false + replicaCount: 1 + autoscaling: + # -- Creates a HorizontalPodAutoscaler for the deployment + enabled: false + minReplicas: 1 + maxReplicas: 10 + targetCPUUtilizationPercentage: 80 + targetMemoryUtilizationPercentage: + +secret: + # -- Labels for the Secret + labels: {} + # -- Annotations for the Secret + annotations: {} + +configmap: + # -- If enabled, promtail config will be created as a ConfigMap instead of a secret + enabled: false + +initContainer: [] + # # -- Specifies whether the init container for setting inotify max user instances is to be enabled + # - name: init + # # -- Docker registry, image and tag for the init container image + # image: docker.io/busybox:1.33 + # # -- Docker image pull policy for the init container image + # imagePullPolicy: IfNotPresent + # # -- The inotify max user instances to configure + # command: + # - sh + # - -c + # - sysctl -w fs.inotify.max_user_instances=128 + # securityContext: + # privileged: true + +image: + # -- The Docker registry + registry: docker.io + # -- Docker image repository + repository: grafana/promtail + # -- Overrides the image tag whose default is the chart's appVersion + tag: null + # -- Docker image pull policy + pullPolicy: IfNotPresent + +# -- Image pull secrets for Docker images +imagePullSecrets: [] + +# -- Annotations for the DaemonSet +annotations: + ignore-check.kube-linter.io/run-as-non-root: "This deployment needs to run as root user to modify log files" + ignore-check.kube-linter.io/writable-host-mount: "This deployment needs writable volume mount on host to capture logs" + +# -- The update strategy for the DaemonSet +updateStrategy: + type: RollingUpdate + +# -- Pod labels +podLabels: {} + +# -- Pod annotations +podAnnotations: {} +# prometheus.io/scrape: "true" +# prometheus.io/port: "http-metrics" + +# -- The name of the PriorityClass +priorityClassName: null + +# -- Liveness probe +livenessProbe: {} + +# -- Readiness probe +# @default -- See `values.yaml` +readinessProbe: + failureThreshold: 5 + httpGet: + path: "{{ printf `%s/ready` .Values.httpPathPrefix }}" + port: http-metrics + initialDelaySeconds: 10 + periodSeconds: 10 + successThreshold: 1 + timeoutSeconds: 1 + +# -- Resource requests and limits +resources: + limits: + cpu: 200m + memory: 128Mi + requests: + cpu: 100m + memory: 128Mi + +# -- The security context for pods +podSecurityContext: + runAsUser: 0 + runAsGroup: 0 + +# -- The security context for containers +containerSecurityContext: + readOnlyRootFilesystem: true + capabilities: + drop: + - ALL + allowPrivilegeEscalation: false + +rbac: + # -- Specifies whether RBAC resources are to be created + create: true + # -- Specifies whether a PodSecurityPolicy is to be created + pspEnabled: false + +# -- The name of the Namespace to deploy +# If not set, `.Release.Namespace` is used +namespace: null + +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: null + # -- Image pull secrets for the service account + imagePullSecrets: [] + # -- Annotations for the service account + annotations: {} + +# -- Node selector for pods +nodeSelector: {} + +# -- Affinity configuration for pods +affinity: {} + +# -- Tolerations for pods. By default, pods will be scheduled on master/control-plane nodes. +tolerations: + - key: node-role.kubernetes.io/master + operator: Exists + effect: NoSchedule + - key: node-role.kubernetes.io/control-plane + operator: Exists + effect: NoSchedule + +# -- Default volumes that are mounted into pods. In most cases, these should not be changed. +# Use `extraVolumes`/`extraVolumeMounts` for additional custom volumes. +# @default -- See `values.yaml` +defaultVolumes: + - name: run + hostPath: + path: /run/promtail + - name: containers + hostPath: + path: /var/lib/docker/containers + - name: pods + hostPath: + path: /var/log/pods + +# -- Default volume mounts. Corresponds to `volumes`. +# @default -- See `values.yaml` +defaultVolumeMounts: + - name: run + mountPath: /run/promtail + - name: containers + mountPath: /var/lib/docker/containers + readOnly: true + - name: pods + mountPath: /var/log/pods + readOnly: true + +# Extra volumes to be added in addition to those specified under `defaultVolumes`. +extraVolumes: [] + +# Extra volume mounts together. Corresponds to `extraVolumes`. +extraVolumeMounts: [] + +# Extra args for the Promtail container. +extraArgs: [] +# -- Example: +# -- extraArgs: +# -- - -client.external-labels=hostname=$(HOSTNAME) + +# -- Extra environment variables +extraEnv: [] + +# -- Extra environment variables from secrets or configmaps +extraEnvFrom: [] + +# -- Configure enableServiceLinks in pod +enableServiceLinks: true + +# ServiceMonitor configuration +serviceMonitor: + # -- If enabled, ServiceMonitor resources for Prometheus Operator are created + enabled: false + # -- Alternative namespace for ServiceMonitor resources + namespace: null + # -- Namespace selector for ServiceMonitor resources + namespaceSelector: {} + # -- ServiceMonitor annotations + annotations: {} + # -- Additional ServiceMonitor labels + labels: {} + # -- ServiceMonitor scrape interval + interval: null + # -- ServiceMonitor scrape timeout in Go duration format (e.g. 15s) + scrapeTimeout: null + # -- ServiceMonitor relabel configs to apply to samples before scraping + # https://github.com/prometheus-operator/prometheus-operator/blob/master/Documentation/api.md#relabelconfig + # (defines `relabel_configs`) + relabelings: [] + # -- ServiceMonitor relabel configs to apply to samples as the last + # step before ingestion + # https://github.com/prometheus-operator/prometheus-operator/blob/master/Documentation/api.md#relabelconfig + # (defines `metric_relabel_configs`) + metricRelabelings: [] + # --ServiceMonitor will add labels from the service to the Prometheus metric + # https://github.com/prometheus-operator/prometheus-operator/blob/main/Documentation/api.md#servicemonitorspec + targetLabels: [] + # -- ServiceMonitor will use http by default, but you can pick https as well + scheme: http + # -- ServiceMonitor will use these tlsConfig settings to make the health check requests + tlsConfig: null + # -- Prometheus rules will be deployed for alerting purposes + prometheusRule: + enabled: false + additionalLabels: {} + # namespace: + rules: [] + # - alert: PromtailRequestErrors + # expr: 100 * sum(rate(promtail_request_duration_seconds_count{status_code=~"5..|failed"}[1m])) by (namespace, job, route, instance) / sum(rate(promtail_request_duration_seconds_count[1m])) by (namespace, job, route, instance) > 10 + # for: 5m + # labels: + # severity: critical + # annotations: + # description: | + # The {{ $labels.job }} {{ $labels.route }} is experiencing + # {{ printf \"%.2f\" $value }} errors. + # VALUE = {{ $value }} + # LABELS = {{ $labels }} + # summary: Promtail request errors (instance {{ $labels.instance }}) + # - alert: PromtailRequestLatency + # expr: histogram_quantile(0.99, sum(rate(promtail_request_duration_seconds_bucket[5m])) by (le)) > 1 + # for: 5m + # labels: + # severity: critical + # annotations: + # summary: Promtail request latency (instance {{ $labels.instance }}) + # description: | + # The {{ $labels.job }} {{ $labels.route }} is experiencing + # {{ printf \"%.2f\" $value }}s 99th percentile latency. + # VALUE = {{ $value }} + # LABELS = {{ $labels }} + +# Extra containers created as part of a Promtail Deployment resource +# - spec for Container: +# https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.23/#container-v1-core +# +# Note that the key is used as the `name` field, i.e. below will create a +# container named `promtail-proxy`. +extraContainers: {} + # promtail-proxy: + # image: nginx + # ... + +# -- Configure additional ports and services. For each configured port, a corresponding service is created. +# See values.yaml for details +extraPorts: {} +# syslog: +# name: tcp-syslog +# containerPort: 1514 +# protocol: TCP +# service: +# type: ClusterIP +# clusterIP: null +# port: 1514 +# externalIPs: [] +# nodePort: null +# annotations: {} +# labels: {} +# loadBalancerIP: null +# loadBalancerSourceRanges: [] +# externalTrafficPolicy: null + +# -- PodSecurityPolicy configuration. +# @default -- See `values.yaml` +podSecurityPolicy: + privileged: true + allowPrivilegeEscalation: true + volumes: + - 'secret' + - 'hostPath' + - 'downwardAPI' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'RunAsAny' + fsGroup: + rule: 'RunAsAny' + readOnlyRootFilesystem: true + requiredDropCapabilities: + - ALL + +# -- Section for crafting Promtails config file. The only directly relevant value is `config.file` +# which is a templated string that references the other values and snippets below this key. +# @default -- See `values.yaml` +config: + # -- The log level of the Promtail server + # Must be reference in `config.file` to configure `server.log_level` + # See default config in `values.yaml` + logLevel: info + # -- The port of the Promtail server + # Must be reference in `config.file` to configure `server.http_listen_port` + # See default config in `values.yaml` + serverPort: 3101 + # -- The config of clients of the Promtail server + # Must be reference in `config.file` to configure `clients` + # @default -- See `values.yaml` + clients: + - url: http://{{ .Release.Name }}-loki:3100/loki/api/v1/push + # -- A section of reusable snippets that can be reference in `config.file`. + # Custom snippets may be added in order to reduce redundancy. + # This is especially helpful when multiple `kubernetes_sd_configs` are use which usually have large parts in common. + # @default -- See `values.yaml` + snippets: + pipelineStages: + - cri: {} + - match: + selector: '{app="k8s-triliovault"}' + stages: + - json: + expressions: + level: level + service_type: service_type + pvc_name: pvc_name + transaction_type: transaction_type + transaction_resource_name: transaction_resource_name + transaction_resource_namespace: transaction_resource_namespace + child_transaction_type: child_transaction_type + child_transaction_resource_name: child_transaction_resource_name + child_transaction_resource_namespace: child_transaction_resource_namespace + tvk_instance_id: tvk_instance_id + - labels: + level: + service_type: + pvc_name: + transaction_type: + transaction_resource_name: + transaction_resource_namespace: + child_transaction_type: + child_transaction_resource_name: + child_transaction_resource_namespace: + tvk_instance_id: + common: + - action: replace + source_labels: + - __meta_kubernetes_pod_node_name + target_label: node_name + - action: replace + source_labels: + - __meta_kubernetes_namespace + target_label: namespace + - action: replace + replacement: $1 + separator: / + source_labels: + - namespace + - app + target_label: job + - action: replace + source_labels: + - __meta_kubernetes_pod_name + target_label: pod + - action: replace + source_labels: + - __meta_kubernetes_pod_container_name + target_label: container + - action: replace + replacement: /var/log/pods/*$1/*.log + separator: / + source_labels: + - __meta_kubernetes_pod_uid + - __meta_kubernetes_pod_container_name + target_label: __path__ + - action: replace + replacement: /var/log/pods/*$1/*.log + regex: true/(.*) + separator: / + source_labels: + - __meta_kubernetes_pod_annotationpresent_kubernetes_io_config_hash + - __meta_kubernetes_pod_annotation_kubernetes_io_config_hash + - __meta_kubernetes_pod_container_name + target_label: __path__ + + # If set to true, adds an additional label for the scrape job. + # This helps debug the Promtail config. + addScrapeJobLabel: false + + # -- You can put here any keys that will be directly added to the config file's 'limits_config' block. + # @default -- empty + extraLimitsConfig: "" + + # -- You can put here any keys that will be directly added to the config file's 'server' block. + # @default -- empty + extraServerConfigs: "" + + # -- You can put here any additional scrape configs you want to add to the config file. + # @default -- empty + extraScrapeConfigs: "" + + # -- You can put here any additional relabel_configs to "kubernetes-pods" job + extraRelabelConfigs: [] + + scrapeConfigs: | + # See also https://github.com/grafana/loki/blob/master/production/ksonnet/promtail/scrape_config.libsonnet for reference + - job_name: kubernetes-pods + pipeline_stages: + {{- toYaml .Values.config.snippets.pipelineStages | nindent 4 }} + kubernetes_sd_configs: + - role: pod + relabel_configs: + - source_labels: + - __meta_kubernetes_pod_controller_name + regex: ([0-9a-z-.]+?)(-[0-9a-f]{8,10})? + action: replace + target_label: __tmp_controller_name + - source_labels: + - __meta_kubernetes_pod_label_app_kubernetes_io_name + - __meta_kubernetes_pod_label_app + - __tmp_controller_name + - __meta_kubernetes_pod_name + regex: ^;*([^;]+)(;.*)?$ + action: replace + target_label: app + - source_labels: + - __meta_kubernetes_pod_label_app_kubernetes_io_instance + - __meta_kubernetes_pod_label_release + regex: ^;*([^;]+)(;.*)?$ + action: replace + target_label: instance + - source_labels: + - __meta_kubernetes_pod_label_app_kubernetes_io_component + - __meta_kubernetes_pod_label_component + regex: ^;*([^;]+)(;.*)?$ + action: replace + target_label: component + - action: keep + source_labels: + - app + - instance + - component + regex: '(k8s-trilio).*' + {{- if .Values.config.snippets.addScrapeJobLabel }} + - replacement: kubernetes-pods + target_label: scrape_job + {{- end }} + {{- toYaml .Values.config.snippets.common | nindent 4 }} + {{- with .Values.config.snippets.extraRelabelConfigs }} + {{- toYaml . | nindent 4 }} + {{- end }} + + # -- Config file contents for Promtail. + # Must be configured as string. + # It is templated so it can be assembled from reusable snippets in order to avoid redundancy. + # @default -- See `values.yaml` + file: | + server: + log_level: {{ .Values.config.logLevel }} + http_listen_port: {{ .Values.config.serverPort }} + {{- with .Values.httpPathPrefix }} + http_path_prefix: {{ . }} + {{- end }} + {{- tpl .Values.config.snippets.extraServerConfigs . | nindent 2 }} + + clients: + {{- tpl (toYaml .Values.config.clients) . | nindent 2 }} + + positions: + filename: /run/promtail/positions.yaml + + scrape_configs: + {{- tpl .Values.config.snippets.scrapeConfigs . | nindent 2 }} + {{- tpl .Values.config.snippets.extraScrapeConfigs . | nindent 2 }} + + limits_config: + {{- tpl .Values.config.snippets.extraLimitsConfig . | nindent 2 }} + +networkPolicy: + # -- Specifies whether Network Policies should be created + enabled: false + metrics: + # -- Specifies the Pods which are allowed to access the metrics port. + # As this is cross-namespace communication, you also neeed the namespaceSelector. + podSelector: {} + # -- Specifies the namespaces which are allowed to access the metrics port + namespaceSelector: {} + # -- Specifies specific network CIDRs which are allowed to access the metrics port. + # In case you use namespaceSelector, you also have to specify your kubelet networks here. + # The metrics ports are also used for probes. + cidrs: [] + k8sApi: + # -- Specify the k8s API endpoint port + port: 8443 + # -- Specifies specific network CIDRs you want to limit access to + cidrs: [] + +# -- Base path to server all API routes fro +httpPathPrefix: "" + +# -- Extra K8s manifests to deploy +extraObjects: [] + # - apiVersion: "kubernetes-client.io/v1" + # kind: ExternalSecret + # metadata: + # name: promtail-secrets + # spec: + # backendType: gcpSecretsManager + # data: + # - key: promtail-oauth2-creds + # name: client_secret diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/_helpers.tpl new file mode 100644 index 000000000..9fb468f83 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/_helpers.tpl @@ -0,0 +1,50 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "logging.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 "logging.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 "logging.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +The service name to connect to Loki. Defaults to the same logic as "loki.fullname" +*/}} +{{- define "loki.serviceName" -}} +{{- if .Values.loki.serviceName -}} +{{- .Values.loki.serviceName -}} +{{- else if .Values.loki.fullnameOverride -}} +{{- .Values.loki.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default "loki" .Values.loki.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- .Release.Name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/datasources.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/datasources.yaml new file mode 100644 index 000000000..f5ca78f23 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/logging/templates/datasources.yaml @@ -0,0 +1,24 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "logging.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + app: {{ template "logging.name" . }} + chart: {{ template "logging.chart" . }} + release: {{ .Release.Name }} + heritage: {{ .Release.Service }} + app.kubernetes.io/instance: {{ template "logging.name" . }} + {{- include "k8s-triliovault-operator.observability" . | nindent 4 }} + grafana_datasource: "1" +data: + logging-datasource.yaml: |- + apiVersion: 1 + datasources: +{{- if .Values.loki.enabled }} + - name: Loki + type: loki + access: proxy + url: http://{{(include "loki.serviceName" .)}}:{{ .Values.loki.service.port }} + version: 1 +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/Chart.yaml new file mode 100644 index 000000000..13ffa6d15 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/Chart.yaml @@ -0,0 +1,16 @@ +apiVersion: v2 +appVersion: 0.1.0 +dependencies: +- condition: prometheus.enabled + name: prometheus + repository: https://prometheus-community.github.io/helm-charts + version: ^15.8.7 +description: Monitoring Stack designed to manage the K8s-TrilioVault Application's + Monitoring. +icon: https://www.trilio.io/wp-content/uploads/2021/01/Trilio-2020-logo-RGB-gray-green.png +kubeVersion: '>=1.19.0-0' +maintainers: +- email: support@trilio.io + name: Trilio +name: monitoring +version: 0.1.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/Chart.yaml new file mode 100644 index 000000000..609fb7386 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/Chart.yaml @@ -0,0 +1,22 @@ +apiVersion: v2 +appVersion: 2.34.0 +dependencies: +- condition: kubeStateMetrics.enabled + name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 4.7.* +description: Prometheus is a monitoring system and time series database. +home: https://prometheus.io/ +icon: https://raw.githubusercontent.com/prometheus/prometheus.github.io/master/assets/prometheus_logo-cb55bb5c346.png +maintainers: +- email: support@trilio.io + name: Trilio +name: prometheus +sources: +- https://github.com/prometheus/alertmanager +- https://github.com/prometheus/prometheus +- https://github.com/prometheus/pushgateway +- https://github.com/prometheus/node_exporter +- https://github.com/kubernetes/kube-state-metrics +type: application +version: 15.8.7 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/Chart.yaml new file mode 100644 index 000000000..83d0685a1 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +appVersion: 2.4.1 +description: Install kube-state-metrics to generate and expose cluster-level metrics +home: https://github.com/kubernetes/kube-state-metrics/ +keywords: +- metric +- monitoring +- prometheus +- kubernetes +maintainers: +- email: support@trilio.io + name: Trilio +name: kube-state-metrics +sources: +- https://github.com/kubernetes/kube-state-metrics/ +type: application +version: 4.7.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt new file mode 100644 index 000000000..5a646e0cc --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt @@ -0,0 +1,10 @@ +kube-state-metrics is a simple service that listens to the Kubernetes API server and generates metrics about the state of the objects. +The exposed metrics can be found here: +https://github.com/kubernetes/kube-state-metrics/blob/master/docs/README.md#exposed-metrics + +The metrics are exported on the HTTP endpoint /metrics on the listening port. +In your case, {{ template "kube-state-metrics.fullname" . }}.{{ template "kube-state-metrics.namespace" . }}.svc.cluster.local:{{ .Values.service.port }}/metrics + +They are served either as plaintext or protobuf depending on the Accept header. +They are designed to be consumed either by Prometheus itself or by a scraper that is compatible with scraping a Prometheus client endpoint. + diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl new file mode 100644 index 000000000..976b27337 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl @@ -0,0 +1,82 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "kube-state-metrics.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 "kube-state-metrics.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 the name of the service account to use +*/}} +{{- define "kube-state-metrics.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "kube-state-metrics.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts +*/}} +{{- define "kube-state-metrics.namespace" -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "kube-state-metrics.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Generate basic labels +*/}} +{{- define "kube-state-metrics.labels" }} +helm.sh/chart: {{ template "kube-state-metrics.chart" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/component: metrics +app.kubernetes.io/part-of: {{ template "kube-state-metrics.name" . }} +{{- include "kube-state-metrics.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +{{- if .Values.customLabels }} +{{ toYaml .Values.customLabels }} +{{- end }} +{{- if .Values.releaseLabel }} +release: {{ .Release.Name }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "kube-state-metrics.selectorLabels" }} +app.kubernetes.io/name: {{ include "kube-state-metrics.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..cf9f628d0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.rbac.create .Values.rbac.useClusterRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole +{{- if .Values.rbac.useExistingRole }} + name: {{ .Values.rbac.useExistingRole }} +{{- else }} + name: {{ template "kube-state-metrics.fullname" . }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml new file mode 100644 index 000000000..ac5387f10 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml @@ -0,0 +1,156 @@ +apiVersion: apps/v1 +{{- if .Values.autosharding.enabled }} +kind: StatefulSet +{{- else }} +kind: Deployment +{{- end }} +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +spec: + selector: + matchLabels: + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + replicas: {{ .Values.replicas }} + {{- if .Values.autosharding.enabled }} + serviceName: {{ template "kube-state-metrics.fullname" . }} + updateStrategy: + type: RollingUpdate + volumeClaimTemplates: [] + {{- else }} + strategy: + type: RollingUpdate + {{- end }} + template: + metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 8 }} + {{- if .Values.podAnnotations }} + annotations: +{{ toYaml .Values.podAnnotations | indent 8 }} + {{- end }} + spec: + hostNetwork: {{ .Values.hostNetwork }} + serviceAccountName: {{ template "kube-state-metrics.serviceAccountName" . }} + {{- if .Values.securityContext.enabled }} + securityContext: + fsGroup: {{ .Values.securityContext.fsGroup }} + runAsGroup: {{ .Values.securityContext.runAsGroup }} + runAsUser: {{ .Values.securityContext.runAsUser }} + {{- end }} + {{- if .Values.priorityClassName }} + priorityClassName: {{ .Values.priorityClassName }} + {{- end }} + containers: + - name: {{ .Chart.Name }} + {{- if .Values.autosharding.enabled }} + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + {{- end }} + args: + {{- if .Values.extraArgs }} + {{- range .Values.extraArgs }} + - {{ . }} + {{- end }} + {{- end }} + {{- if .Values.service.port }} + - --port={{ .Values.service.port | default 8080}} + {{- end }} + {{- if .Values.collectors }} + - --resources={{ .Values.collectors | join "," }} + {{- end }} + {{- if .Values.metricLabelsAllowlist }} + - --metric-labels-allowlist={{ .Values.metricLabelsAllowlist | join "," }} + {{- end }} + {{- if .Values.metricAnnotationsAllowList }} + - --metric-annotations-allowlist={{ .Values.metricAnnotationsAllowList | join "," }} + {{- end }} + {{- if .Values.metricAllowlist }} + - --metric-allowlist={{ .Values.metricAllowlist | join "," }} + {{- end }} + {{- if .Values.metricDenylist }} + - --metric-denylist={{ .Values.metricDenylist | join "," }} + {{- end }} + {{- if .Values.namespaces }} + - --namespaces={{ tpl (.Values.namespaces | join ",") $ }} + {{- end }} + {{- if .Values.namespacesDenylist }} + - --namespaces-denylist={{ tpl (.Values.namespacesDenylist | join ",") $ }} + {{- end }} + {{- if .Values.autosharding.enabled }} + - --pod=$(POD_NAME) + - --pod-namespace=$(POD_NAMESPACE) + {{- end }} + {{- if .Values.kubeconfig.enabled }} + - --kubeconfig=/opt/k8s/.kube/config + {{- end }} + {{- if .Values.selfMonitor.telemetryHost }} + - --telemetry-host={{ .Values.selfMonitor.telemetryHost }} + {{- end }} + - --telemetry-port={{ .Values.selfMonitor.telemetryPort | default 8081 }} + {{- if .Values.kubeconfig.enabled }} + volumeMounts: + - name: kubeconfig + mountPath: /opt/k8s/.kube/ + readOnly: true + {{- end }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + image: "{{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}" + ports: + - containerPort: {{ .Values.service.port | default 8080}} + name: "http" + {{- if .Values.selfMonitor.enabled }} + - containerPort: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + name: "metrics" + {{- end }} + livenessProbe: + httpGet: + path: /healthz + port: {{ .Values.service.port | default 8080}} + initialDelaySeconds: 5 + timeoutSeconds: 5 + readinessProbe: + httpGet: + path: / + port: {{ .Values.service.port | default 8080}} + initialDelaySeconds: 5 + timeoutSeconds: 5 + {{- if .Values.resources }} + resources: +{{ toYaml .Values.resources | indent 10 }} +{{- end }} +{{- if .Values.containerSecurityContext }} + securityContext: +{{ toYaml .Values.containerSecurityContext | indent 10 }} +{{- end }} +{{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.affinity }} + affinity: +{{ toYaml .Values.affinity | indent 8 }} + {{- end }} + {{- if .Values.nodeSelector }} + nodeSelector: +{{ toYaml .Values.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: +{{ toYaml .Values.tolerations | indent 8 }} + {{- end }} + {{- if .Values.kubeconfig.enabled}} + volumes: + - name: kubeconfig + secret: + secretName: {{ template "kube-state-metrics.fullname" . }}-kubeconfig + {{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml new file mode 100644 index 000000000..6af008450 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml @@ -0,0 +1,12 @@ +{{- if .Values.kubeconfig.enabled -}} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "kube-state-metrics.fullname" . }}-kubeconfig + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +type: Opaque +data: + config: '{{ .Values.kubeconfig.secret }}' +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml new file mode 100644 index 000000000..cbcf3a37e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.podDisruptionBudget -}} +apiVersion: policy/v1beta1 +kind: PodDisruptionBudget +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +spec: + selector: + matchLabels: + app.kubernetes.io/name: {{ template "kube-state-metrics.name" . }} +{{ toYaml .Values.podDisruptionBudget | indent 2 }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml new file mode 100644 index 000000000..3299056ab --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml @@ -0,0 +1,39 @@ +{{- if .Values.podSecurityPolicy.enabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +{{- if .Values.podSecurityPolicy.annotations }} + annotations: +{{ toYaml .Values.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + volumes: + - 'secret' +{{- if .Values.podSecurityPolicy.additionalVolumes }} +{{ toYaml .Values.podSecurityPolicy.additionalVolumes | indent 4 }} +{{- end }} + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'MustRunAsNonRoot' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml new file mode 100644 index 000000000..69047d4ff --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.podSecurityPolicy.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: psp-{{ template "kube-state-metrics.fullname" . }} +rules: +{{- $kubeTargetVersion := default .Capabilities.KubeVersion.GitVersion .Values.kubeTargetVersionOverride }} +{{- if semverCompare "> 1.15.0-0" $kubeTargetVersion }} +- apiGroups: ['policy'] +{{- else }} +- apiGroups: ['extensions'] +{{- end }} + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ template "kube-state-metrics.fullname" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml new file mode 100644 index 000000000..03c56d575 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.podSecurityPolicy.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: psp-{{ template "kube-state-metrics.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: psp-{{ template "kube-state-metrics.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/role.yaml new file mode 100644 index 000000000..e514e3c01 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/role.yaml @@ -0,0 +1,187 @@ +{{- if and (eq .Values.rbac.create true) (not .Values.rbac.useExistingRole) -}} +{{- range (ternary (split "," .Values.namespaces) (list "") (eq $.Values.rbac.useClusterRole false)) }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +{{- if eq $.Values.rbac.useClusterRole false }} +kind: Role +{{- else }} +kind: ClusterRole +{{- end }} +metadata: + labels: + {{- include "kube-state-metrics.labels" $ | indent 4 }} + name: {{ template "kube-state-metrics.fullname" $ }} +{{- if eq $.Values.rbac.useClusterRole false }} + namespace: {{ . }} +{{- end }} +rules: +{{ if has "certificatesigningrequests" $.Values.collectors }} +- apiGroups: ["certificates.k8s.io"] + resources: + - certificatesigningrequests + verbs: ["list", "watch"] +{{ end -}} +{{ if has "configmaps" $.Values.collectors }} +- apiGroups: [""] + resources: + - configmaps + verbs: ["list", "watch"] +{{ end -}} +{{ if has "cronjobs" $.Values.collectors }} +- apiGroups: ["batch"] + resources: + - cronjobs + verbs: ["list", "watch"] +{{ end -}} +{{ if has "daemonsets" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - daemonsets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "deployments" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - deployments + verbs: ["list", "watch"] +{{ end -}} +{{ if has "endpoints" $.Values.collectors }} +- apiGroups: [""] + resources: + - endpoints + verbs: ["list", "watch"] +{{ end -}} +{{ if has "horizontalpodautoscalers" $.Values.collectors }} +- apiGroups: ["autoscaling"] + resources: + - horizontalpodautoscalers + verbs: ["list", "watch"] +{{ end -}} +{{ if has "ingresses" $.Values.collectors }} +- apiGroups: ["extensions", "networking.k8s.io"] + resources: + - ingresses + verbs: ["list", "watch"] +{{ end -}} +{{ if has "jobs" $.Values.collectors }} +- apiGroups: ["batch"] + resources: + - jobs + verbs: ["list", "watch"] +{{ end -}} +{{ if has "limitranges" $.Values.collectors }} +- apiGroups: [""] + resources: + - limitranges + verbs: ["list", "watch"] +{{ end -}} +{{ if has "mutatingwebhookconfigurations" $.Values.collectors }} +- apiGroups: ["admissionregistration.k8s.io"] + resources: + - mutatingwebhookconfigurations + verbs: ["list", "watch"] +{{ end -}} +{{ if has "namespaces" $.Values.collectors }} +- apiGroups: [""] + resources: + - namespaces + verbs: ["list", "watch"] +{{ end -}} +{{ if has "networkpolicies" $.Values.collectors }} +- apiGroups: ["networking.k8s.io"] + resources: + - networkpolicies + verbs: ["list", "watch"] +{{ end -}} +{{ if has "nodes" $.Values.collectors }} +- apiGroups: [""] + resources: + - nodes + verbs: ["list", "watch"] +{{ end -}} +{{ if has "persistentvolumeclaims" $.Values.collectors }} +- apiGroups: [""] + resources: + - persistentvolumeclaims + verbs: ["list", "watch"] +{{ end -}} +{{ if has "persistentvolumes" $.Values.collectors }} +- apiGroups: [""] + resources: + - persistentvolumes + verbs: ["list", "watch"] +{{ end -}} +{{ if has "poddisruptionbudgets" $.Values.collectors }} +- apiGroups: ["policy"] + resources: + - poddisruptionbudgets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "pods" $.Values.collectors }} +- apiGroups: [""] + resources: + - pods + verbs: ["list", "watch"] +{{ end -}} +{{ if has "replicasets" $.Values.collectors }} +- apiGroups: ["extensions", "apps"] + resources: + - replicasets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "replicationcontrollers" $.Values.collectors }} +- apiGroups: [""] + resources: + - replicationcontrollers + verbs: ["list", "watch"] +{{ end -}} +{{ if has "resourcequotas" $.Values.collectors }} +- apiGroups: [""] + resources: + - resourcequotas + verbs: ["list", "watch"] +{{ end -}} +{{ if has "secrets" $.Values.collectors }} +- apiGroups: [""] + resources: + - secrets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "services" $.Values.collectors }} +- apiGroups: [""] + resources: + - services + verbs: ["list", "watch"] +{{ end -}} +{{ if has "statefulsets" $.Values.collectors }} +- apiGroups: ["apps"] + resources: + - statefulsets + verbs: ["list", "watch"] +{{ end -}} +{{ if has "storageclasses" $.Values.collectors }} +- apiGroups: ["storage.k8s.io"] + resources: + - storageclasses + verbs: ["list", "watch"] +{{ end -}} +{{ if has "validatingwebhookconfigurations" $.Values.collectors }} +- apiGroups: ["admissionregistration.k8s.io"] + resources: + - validatingwebhookconfigurations + verbs: ["list", "watch"] +{{ end -}} +{{ if has "volumeattachments" $.Values.collectors }} +- apiGroups: ["storage.k8s.io"] + resources: + - volumeattachments + verbs: ["list", "watch"] +{{ end -}} +{{ if has "verticalpodautoscalers" $.Values.collectors }} +- apiGroups: ["autoscaling.k8s.io"] + resources: + - verticalpodautoscalers + verbs: ["list", "watch"] +{{ end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml new file mode 100644 index 000000000..135094f7b --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml @@ -0,0 +1,24 @@ +{{- if and (eq .Values.rbac.create true) (eq .Values.rbac.useClusterRole false) -}} +{{- range (split "," $.Values.namespaces) }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + {{- include "kube-state-metrics.labels" $ | indent 4 }} + name: {{ template "kube-state-metrics.fullname" $ }} + namespace: {{ . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role +{{- if (not $.Values.rbac.useExistingRole) }} + name: {{ template "kube-state-metrics.fullname" $ }} +{{- else }} + name: {{ $.Values.rbac.useExistingRole }} +{{- end }} +subjects: +- kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" $ }} + namespace: {{ template "kube-state-metrics.namespace" $ }} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/service.yaml new file mode 100644 index 000000000..5a2d8eab0 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/service.yaml @@ -0,0 +1,38 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + annotations: + {{- if .Values.prometheusScrape }} + prometheus.io/scrape: '{{ .Values.prometheusScrape }}' + {{- end }} + {{- if .Values.service.annotations }} + {{- toYaml .Values.service.annotations | nindent 4 }} + {{- end }} +spec: + type: "{{ .Values.service.type }}" + ports: + - name: "http" + protocol: TCP + port: {{ .Values.service.port | default 8080}} + {{- if .Values.service.nodePort }} + nodePort: {{ .Values.service.nodePort }} + {{- end }} + targetPort: {{ .Values.service.port | default 8080}} + {{ if .Values.selfMonitor.enabled }} + - name: "metrics" + protocol: TCP + port: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + targetPort: {{ .Values.selfMonitor.telemetryPort | default 8081 }} + {{ end }} +{{- if .Values.service.loadBalancerIP }} + loadBalancerIP: "{{ .Values.service.loadBalancerIP }}" +{{- end }} +{{- if .Values.service.clusterIP }} + clusterIP: "{{ .Values.service.clusterIP }}" +{{- end }} + selector: + {{- include "kube-state-metrics.selectorLabels" . | indent 4 }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml new file mode 100644 index 000000000..e1229eb95 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml @@ -0,0 +1,15 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- if .Values.serviceAccount.annotations }} + annotations: +{{ toYaml .Values.serviceAccount.annotations | indent 4 }} +{{- end }} +imagePullSecrets: +{{ toYaml .Values.serviceAccount.imagePullSecrets | indent 2 }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml new file mode 100644 index 000000000..93a5870f6 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml @@ -0,0 +1,66 @@ +{{- if .Values.prometheus.monitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} + {{- with .Values.prometheus.monitor.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + jobLabel: {{ default "app.kubernetes.io/name" .Values.prometheus.monitor.jobLabel }} + selector: + matchLabels: + {{- if .Values.prometheus.monitor.selectorOverride -}} + {{ toYaml .Values.prometheus.monitor.selectorOverride | nindent 6 }} + {{ else }} + {{- include "kube-state-metrics.selectorLabels" . | indent 6 }} + {{- end }} + endpoints: + - port: http + {{- if .Values.prometheus.monitor.interval }} + interval: {{ .Values.prometheus.monitor.interval }} + {{- end }} + {{- if .Values.prometheus.monitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.prometheus.monitor.scrapeTimeout }} + {{- end }} + {{- if .Values.prometheus.monitor.proxyUrl }} + proxyUrl: {{ .Values.prometheus.monitor.proxyUrl}} + {{- end }} + {{- if .Values.prometheus.monitor.honorLabels }} + honorLabels: true + {{- end }} + {{- if .Values.prometheus.monitor.metricRelabelings }} + metricRelabelings: + {{- toYaml .Values.prometheus.monitor.metricRelabelings | nindent 8 }} + {{- end }} + {{- if .Values.prometheus.monitor.relabelings }} + relabelings: + {{- toYaml .Values.prometheus.monitor.relabelings | nindent 8 }} + {{- end }} + {{- if .Values.selfMonitor.enabled }} + - port: metrics + {{- if .Values.prometheus.monitor.interval }} + interval: {{ .Values.prometheus.monitor.interval }} + {{- end }} + {{- if .Values.prometheus.monitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.prometheus.monitor.scrapeTimeout }} + {{- end }} + {{- if .Values.prometheus.monitor.proxyUrl }} + proxyUrl: {{ .Values.prometheus.monitor.proxyUrl}} + {{- end }} + {{- if .Values.prometheus.monitor.honorLabels }} + honorLabels: true + {{- end }} + {{- if .Values.prometheus.monitor.metricRelabelings }} + metricRelabelings: + {{- toYaml .Values.prometheus.monitor.metricRelabelings | nindent 8 }} + {{- end }} + {{- if .Values.prometheus.monitor.relabelings }} + relabelings: + {{- toYaml .Values.prometheus.monitor.relabelings | nindent 8 }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml new file mode 100644 index 000000000..489de147c --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml @@ -0,0 +1,26 @@ +{{- if and .Values.autosharding.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +rules: +- apiGroups: + - "" + resources: + - pods + verbs: + - get +- apiGroups: + - apps + resourceNames: + - {{ template "kube-state-metrics.fullname" . }} + resources: + - statefulsets + verbs: + - get + - list + - watch +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml new file mode 100644 index 000000000..73b37a4f6 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml @@ -0,0 +1,17 @@ +{{- if and .Values.autosharding.enabled .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} + labels: + {{- include "kube-state-metrics.labels" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: stsdiscovery-{{ template "kube-state-metrics.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "kube-state-metrics.serviceAccountName" . }} + namespace: {{ template "kube-state-metrics.namespace" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/values.yaml new file mode 100644 index 000000000..3feb6b501 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/charts/kube-state-metrics/values.yaml @@ -0,0 +1,233 @@ +# Default values for kube-state-metrics. +prometheusScrape: true +image: + registry: k8s.gcr.io + repository: kube-state-metrics/kube-state-metrics + tag: v2.4.1 + pullPolicy: IfNotPresent + +imagePullSecrets: [] +# - name: "image-pull-secret" + +# If set to true, this will deploy kube-state-metrics as a StatefulSet and the data +# will be automatically sharded across <.Values.replicas> pods using the built-in +# autodiscovery feature: https://github.com/kubernetes/kube-state-metrics#automated-sharding +# This is an experimental feature and there are no stability guarantees. +autosharding: + enabled: false + +replicas: 1 + +# List of additional cli arguments to configure kube-state-metrics +# for example: --enable-gzip-encoding, --log-file, etc. +# all the possible args can be found here: https://github.com/kubernetes/kube-state-metrics/blob/master/docs/cli-arguments.md +extraArgs: [] + +service: + port: 8080 + # Default to clusterIP for backward compatibility + type: ClusterIP + nodePort: 0 + loadBalancerIP: "" + clusterIP: "" + annotations: {} + +## Additional labels to add to all resources +customLabels: {} + # app: kube-state-metrics + +## set to true to add the release label so scraping of the servicemonitor with kube-prometheus-stack works out of the box +releaseLabel: false + +hostNetwork: false + +rbac: + # If true, create & use RBAC resources + create: true + + # Set to a rolename to use existing role - skipping role creating - but still doing serviceaccount and rolebinding to it, rolename set here. + # useExistingRole: your-existing-role + + # If set to false - Run without Cluteradmin privs needed - ONLY works if namespace is also set (if useExistingRole is set this name is used as ClusterRole or Role to bind to) + useClusterRole: true + +serviceAccount: + # Specifies whether a ServiceAccount should be created, require rbac true + 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: + # Reference to one or more secrets to be used when pulling images + # ref: https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/ + imagePullSecrets: [] + # ServiceAccount annotations. + # Use case: AWS EKS IAM roles for service accounts + # ref: https://docs.aws.amazon.com/eks/latest/userguide/specify-service-account-role.html + annotations: {} + +prometheus: + monitor: + enabled: false + additionalLabels: {} + namespace: "" + jobLabel: "" + interval: "" + scrapeTimeout: "" + proxyUrl: "" + selectorOverride: {} + honorLabels: false + metricRelabelings: [] + relabelings: [] + +## Specify if a Pod Security Policy for kube-state-metrics must be created +## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ +## +podSecurityPolicy: + enabled: false + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + additionalVolumes: [] + +securityContext: + enabled: true + runAsGroup: 65534 + runAsUser: 65534 + fsGroup: 65534 + +## Specify security settings for a Container +## Allows overrides and additional options compared to (Pod) securityContext +## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container +containerSecurityContext: {} + +## Node labels for pod assignment +## Ref: https://kubernetes.io/docs/user-guide/node-selection/ +nodeSelector: {} + +## Affinity settings for pod assignment +## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ +affinity: {} + +## Tolerations for pod assignment +## Ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ +tolerations: [] + +# Annotations to be added to the pod +podAnnotations: {} + +## Assign a PriorityClassName to pods if set +# priorityClassName: "" + +# Ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ +podDisruptionBudget: {} + +updateStrategy: + type: RollingUpdate + +# Comma-separated list of metrics to be exposed. +# This list comprises of exact metric names and/or regex patterns. +# The allowlist and denylist are mutually exclusive. +metricAllowlist: [] + +# Comma-separated list of metrics not to be enabled. +# This list comprises of exact metric names and/or regex patterns. +# The allowlist and denylist are mutually exclusive. +metricDenylist: [] + +# Comma-separated list of additional Kubernetes label keys that will be used in the resource's +# labels metric. By default the metric contains only name and namespace labels. +# To include additional labels, provide a list of resource names in their plural form and Kubernetes +# label keys you would like to allow for them (Example: '=namespaces=[k8s-label-1,k8s-label-n,...],pods=[app],...)'. +# A single '*' can be provided per resource instead to allow any labels, but that has +# severe performance implications (Example: '=pods=[*]'). +metricLabelsAllowlist: [] + # - namespaces=[k8s-label-1,k8s-label-n] + +# Comma-separated list of Kubernetes annotations keys that will be used in the resource' +# labels metric. By default the metric contains only name and namespace labels. +# To include additional annotations provide a list of resource names in their plural form and Kubernetes +# annotation keys you would like to allow for them (Example: '=namespaces=[kubernetes.io/team,...],pods=[kubernetes.io/team],...)'. +# A single '*' can be provided per resource instead to allow any annotations, but that has +# severe performance implications (Example: '=pods=[*]'). +metricAnnotationsAllowList: [] + # - pods=[k8s-annotation-1,k8s-annotation-n] + +# Available collectors for kube-state-metrics. +# By default, all available resources are enabled, comment out to disable. +collectors: + - certificatesigningrequests + - configmaps + - cronjobs + - daemonsets + - deployments + - endpoints + - horizontalpodautoscalers + - ingresses + - jobs + - limitranges + - mutatingwebhookconfigurations + - namespaces + - networkpolicies + - nodes + - persistentvolumeclaims + - persistentvolumes + - poddisruptionbudgets + - pods + - replicasets + - replicationcontrollers + - resourcequotas + - secrets + - services + - statefulsets + - storageclasses + - validatingwebhookconfigurations + - volumeattachments + # - verticalpodautoscalers # not a default resource, see also: https://github.com/kubernetes/kube-state-metrics#enabling-verticalpodautoscalers + +# Enabling kubeconfig will pass the --kubeconfig argument to the container +kubeconfig: + enabled: false + # base64 encoded kube-config file + secret: + +# Comma-separated list of namespaces to be enabled for collecting resources. By default all namespaces are collected. +namespaces: "" + +# Comma-separated list of namespaces not to be enabled. If namespaces and namespaces-denylist are both set, +# only namespaces that are excluded in namespaces-denylist will be used. +namespacesDenylist: "" + +## Override the deployment namespace +## +namespaceOverride: "" + +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: 500m + memory: 128Mi + requests: + cpu: 100m + memory: 64Mi + +## Provide a k8s version to define apiGroups for podSecurityPolicy Cluster Role. +## For example: kubeTargetVersionOverride: 1.14.9 +## +kubeTargetVersionOverride: "" + +# Enable self metrics configuration for service and Service Monitor +# Default values for telemetry configuration can be overridden +selfMonitor: + enabled: false + # telemetryHost: 0.0.0.0 + # telemetryPort: 8081 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/NOTES.txt b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/NOTES.txt new file mode 100644 index 000000000..0e8868f0b --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/NOTES.txt @@ -0,0 +1,112 @@ +{{- if .Values.server.enabled -}} +The Prometheus server can be accessed via port {{ .Values.server.service.servicePort }} on the following DNS name from within your cluster: +{{ template "prometheus.server.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if .Values.server.ingress.enabled -}} +From outside the cluster, the server URL(s) are: +{{- range .Values.server.ingress.hosts }} +http://{{ . }} +{{- end }} +{{- else }} +Get the Prometheus server URL by running these commands in the same shell: +{{- if contains "NodePort" .Values.server.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus.server.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.server.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "prometheus.server.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "prometheus.server.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.server.service.servicePort }} +{{- else if contains "ClusterIP" .Values.server.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app={{ template "prometheus.name" . }},component={{ .Values.server.name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9090 +{{- end }} +{{- end }} + +{{- if .Values.server.persistentVolume.enabled }} +{{- else }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the Server pod is terminated. ##### +################################################################################# +{{- end }} +{{- end }} + +{{ if .Values.alertmanager.enabled }} +The Prometheus alertmanager can be accessed via port {{ .Values.alertmanager.service.servicePort }} on the following DNS name from within your cluster: +{{ template "prometheus.alertmanager.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if .Values.alertmanager.ingress.enabled -}} +From outside the cluster, the alertmanager URL(s) are: +{{- range .Values.alertmanager.ingress.hosts }} +http://{{ . }} +{{- end }} +{{- else }} +Get the Alertmanager URL by running these commands in the same shell: +{{- if contains "NodePort" .Values.alertmanager.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus.alertmanager.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.alertmanager.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "prometheus.alertmanager.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "prometheus.alertmanager.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.alertmanager.service.servicePort }} +{{- else if contains "ClusterIP" .Values.alertmanager.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app={{ template "prometheus.name" . }},component={{ .Values.alertmanager.name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9093 +{{- end }} +{{- end }} + +{{- if .Values.alertmanager.persistentVolume.enabled }} +{{- else }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the AlertManager pod is terminated. ##### +################################################################################# +{{- end }} +{{- end }} + +{{- if .Values.nodeExporter.podSecurityPolicy.enabled }} +{{- else }} +################################################################################# +###### WARNING: Pod Security Policy has been moved to a global property. ##### +###### use .Values.podSecurityPolicy.enabled with pod-based ##### +###### annotations ##### +###### (e.g. .Values.nodeExporter.podSecurityPolicy.annotations) ##### +################################################################################# +{{- end }} + +{{ if .Values.pushgateway.enabled }} +The Prometheus PushGateway can be accessed via port {{ .Values.pushgateway.service.servicePort }} on the following DNS name from within your cluster: +{{ template "prometheus.pushgateway.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local + +{{ if .Values.pushgateway.ingress.enabled -}} +From outside the cluster, the pushgateway URL(s) are: +{{- range .Values.pushgateway.ingress.hosts }} +http://{{ . }} +{{- end }} +{{- else }} +Get the PushGateway URL by running these commands in the same shell: +{{- if contains "NodePort" .Values.pushgateway.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "prometheus.pushgateway.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo http://$NODE_IP:$NODE_PORT +{{- else if contains "LoadBalancer" .Values.pushgateway.service.type }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "prometheus.pushgateway.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "prometheus.pushgateway.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP:{{ .Values.pushgateway.service.servicePort }} +{{- else if contains "ClusterIP" .Values.pushgateway.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app={{ template "prometheus.name" . }},component={{ .Values.pushgateway.name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl --namespace {{ .Release.Namespace }} port-forward $POD_NAME 9091 +{{- end }} +{{- end }} +{{- end }} + +For more information on running Prometheus, visit: +https://prometheus.io/ diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/_helpers.tpl new file mode 100644 index 000000000..2d93181bb --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/_helpers.tpl @@ -0,0 +1,288 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "prometheus.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "prometheus.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create unified labels for prometheus components +*/}} +{{- define "prometheus.common.matchLabels" -}} +app: {{ template "prometheus.name" . }} +release: {{ .Release.Name }} +{{ include "k8s-triliovault-operator.observability" . }} +{{- end -}} + +{{- define "prometheus.common.metaLabels" -}} +chart: {{ template "prometheus.chart" . }} +heritage: {{ .Release.Service }} +{{ include "k8s-triliovault-operator.observability" . }} +{{- end -}} + +{{- define "prometheus.alertmanager.labels" -}} +{{ include "prometheus.alertmanager.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +{{- end -}} + +{{- define "prometheus.alertmanager.matchLabels" -}} +component: {{ .Values.alertmanager.name | quote }} +app.kubernetes.io/instance: {{ .Values.alertmanager.name | quote }} +{{ include "prometheus.common.matchLabels" . }} +{{- end -}} + +{{- define "prometheus.nodeExporter.labels" -}} +{{ include "prometheus.nodeExporter.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +{{- end -}} + +{{- define "prometheus.nodeExporter.matchLabels" -}} +component: {{ .Values.nodeExporter.name | quote }} +app.kubernetes.io/instance: {{ .Values.nodeExporter.name | quote }} +{{ include "prometheus.common.matchLabels" . }} +{{- end -}} + +{{- define "prometheus.pushgateway.labels" -}} +{{ include "prometheus.pushgateway.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +{{- end -}} + +{{- define "prometheus.pushgateway.matchLabels" -}} +component: {{ .Values.pushgateway.name | quote }} +app.kubernetes.io/instance: {{ .Values.pushgateway.name | quote }} +{{ include "prometheus.common.matchLabels" . }} +{{- end -}} + +{{- define "prometheus.server.labels" -}} +{{ include "prometheus.server.matchLabels" . }} +{{ include "prometheus.common.metaLabels" . }} +{{- end -}} + +{{- define "prometheus.server.matchLabels" -}} +component: {{ .Values.server.name | quote }} +app.kubernetes.io/instance: {{ .Values.server.name | quote }} +{{ include "prometheus.common.matchLabels" . }} +{{- 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 "prometheus.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 a fully qualified alertmanager name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} + +{{- define "prometheus.alertmanager.fullname" -}} +{{- if .Values.alertmanager.fullnameOverride -}} +{{- .Values.alertmanager.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name .Values.alertmanager.name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.alertmanager.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Create a fully qualified node-exporter name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "prometheus.nodeExporter.fullname" -}} +{{- if .Values.nodeExporter.fullnameOverride -}} +{{- .Values.nodeExporter.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name .Values.nodeExporter.name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.nodeExporter.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Create a fully qualified Prometheus server name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "prometheus.server.fullname" -}} +{{- if .Values.server.fullnameOverride -}} +{{- .Values.server.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name .Values.server.name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.server.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Create a fully qualified pushgateway name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "prometheus.pushgateway.fullname" -}} +{{- if .Values.pushgateway.fullnameOverride -}} +{{- .Values.pushgateway.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if contains $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name .Values.pushgateway.name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.pushgateway.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Get KubeVersion removing pre-release information. +*/}} +{{- define "prometheus.kubeVersion" -}} + {{- default .Capabilities.KubeVersion.Version (regexFind "v[0-9]+\\.[0-9]+\\.[0-9]+" .Capabilities.KubeVersion.Version) -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for deployment. +*/}} +{{- define "prometheus.deployment.apiVersion" -}} +{{- print "apps/v1" -}} +{{- end -}} +{{/* +Return the appropriate apiVersion for daemonset. +*/}} +{{- define "prometheus.daemonset.apiVersion" -}} +{{- print "apps/v1" -}} +{{- end -}} +{{/* +Return the appropriate apiVersion for networkpolicy. +*/}} +{{- define "prometheus.networkPolicy.apiVersion" -}} +{{- print "networking.k8s.io/v1" -}} +{{- end -}} +{{/* +Return the appropriate apiVersion for podsecuritypolicy. +*/}} +{{- define "prometheus.podSecurityPolicy.apiVersion" -}} +{{- print "policy/v1beta1" -}} +{{- end -}} +{{/* +Return the appropriate apiVersion for rbac. +*/}} +{{- define "rbac.apiVersion" -}} +{{- if .Capabilities.APIVersions.Has "rbac.authorization.k8s.io/v1" }} +{{- print "rbac.authorization.k8s.io/v1" -}} +{{- else -}} +{{- print "rbac.authorization.k8s.io/v1beta1" -}} +{{- end -}} +{{- end -}} +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "ingress.apiVersion" -}} + {{- if and (.Capabilities.APIVersions.Has "networking.k8s.io/v1") (semverCompare ">= 1.19.x" (include "prometheus.kubeVersion" .)) -}} + {{- print "networking.k8s.io/v1" -}} + {{- else if .Capabilities.APIVersions.Has "networking.k8s.io/v1beta1" -}} + {{- print "networking.k8s.io/v1beta1" -}} + {{- else -}} + {{- print "extensions/v1beta1" -}} + {{- end -}} +{{- end -}} + +{{/* +Return if ingress is stable. +*/}} +{{- define "ingress.isStable" -}} + {{- eq (include "ingress.apiVersion" .) "networking.k8s.io/v1" -}} +{{- end -}} + +{{/* +Return if ingress supports ingressClassName. +*/}} +{{- define "ingress.supportsIngressClassName" -}} + {{- or (eq (include "ingress.isStable" .) "true") (and (eq (include "ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18.x" (include "prometheus.kubeVersion" .))) -}} +{{- end -}} +{{/* +Return if ingress supports pathType. +*/}} +{{- define "ingress.supportsPathType" -}} + {{- or (eq (include "ingress.isStable" .) "true") (and (eq (include "ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18.x" (include "prometheus.kubeVersion" .))) -}} +{{- end -}} + +{{/* +Create the name of the service account to use for the alertmanager component +*/}} +{{- define "prometheus.serviceAccountName.alertmanager" -}} +{{- if .Values.serviceAccounts.alertmanager.create -}} + {{ default (include "prometheus.alertmanager.fullname" .) .Values.serviceAccounts.alertmanager.name }} +{{- else -}} + {{ default "default" .Values.serviceAccounts.alertmanager.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the name of the service account to use for the nodeExporter component +*/}} +{{- define "prometheus.serviceAccountName.nodeExporter" -}} +{{- if .Values.serviceAccounts.nodeExporter.create -}} + {{ default (include "prometheus.nodeExporter.fullname" .) .Values.serviceAccounts.nodeExporter.name }} +{{- else -}} + {{ default "default" .Values.serviceAccounts.nodeExporter.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the name of the service account to use for the pushgateway component +*/}} +{{- define "prometheus.serviceAccountName.pushgateway" -}} +{{- if .Values.serviceAccounts.pushgateway.create -}} + {{ default (include "prometheus.pushgateway.fullname" .) .Values.serviceAccounts.pushgateway.name }} +{{- else -}} + {{ default "default" .Values.serviceAccounts.pushgateway.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the name of the service account to use for the server component +*/}} +{{- define "prometheus.serviceAccountName.server" -}} +{{- if .Values.serviceAccounts.server.create -}} + {{ default (include "prometheus.server.fullname" .) .Values.serviceAccounts.server.name }} +{{- else -}} + {{ default "default" .Values.serviceAccounts.server.name }} +{{- end -}} +{{- end -}} + +{{/* +Define the prometheus.namespace template if set with forceNamespace or .Release.Namespace is set +*/}} +{{- define "prometheus.namespace" -}} +{{- if .Values.forceNamespace -}} +{{ printf "namespace: %s" .Values.forceNamespace }} +{{- else -}} +{{ printf "namespace: %s" .Release.Namespace }} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrole.yaml new file mode 100644 index 000000000..c732ff4e5 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrole.yaml @@ -0,0 +1,21 @@ +{{- if and .Values.alertmanager.enabled .Values.rbac.create .Values.alertmanager.useClusterRole (not .Values.alertmanager.useExistingRole) -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRole +metadata: + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" . }} +rules: +{{- if .Values.podSecurityPolicy.enabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ template "prometheus.alertmanager.fullname" . }} +{{- else }} + [] +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrolebinding.yaml new file mode 100644 index 000000000..6f13e98b5 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.alertmanager.enabled .Values.rbac.create .Values.alertmanager.useClusterRole -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRoleBinding +metadata: + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.alertmanager" . }} +{{ include "prometheus.namespace" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole +{{- if (not .Values.alertmanager.useExistingRole) }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{- else }} + name: {{ .Values.alertmanager.useExistingRole }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/cm.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/cm.yaml new file mode 100644 index 000000000..cb09bf067 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/cm.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.alertmanager.enabled (and (empty .Values.alertmanager.configMapOverrideName) (empty .Values.alertmanager.configFromSecret)) -}} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +data: +{{- $root := . -}} +{{- range $key, $value := .Values.alertmanagerFiles }} + {{- if $key | regexMatch ".*\\.ya?ml$" }} + {{ $key }}: | +{{ toYaml $value | default "{}" | indent 4 }} + {{- else }} + {{ $key }}: {{ toYaml $value | indent 4 }} + {{- end }} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/deploy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/deploy.yaml new file mode 100644 index 000000000..8a51d250a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/deploy.yaml @@ -0,0 +1,208 @@ +{{- if and .Values.alertmanager.enabled (not .Values.alertmanager.statefulSet.enabled) -}} +apiVersion: {{ template "prometheus.deployment.apiVersion" . }} +kind: Deployment +metadata: +{{- if .Values.alertmanager.deploymentAnnotations }} + annotations: + {{ toYaml .Values.alertmanager.deploymentAnnotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + selector: + matchLabels: + {{- include "prometheus.alertmanager.matchLabels" . | nindent 6 }} + replicas: {{ .Values.alertmanager.replicaCount }} + {{- if .Values.alertmanager.strategy }} + strategy: +{{ toYaml .Values.alertmanager.strategy | trim | indent 4 }} + {{ if eq .Values.alertmanager.strategy.type "Recreate" }}rollingUpdate: null{{ end }} +{{- end }} + template: + metadata: + {{- if .Values.alertmanager.podAnnotations }} + annotations: + {{ toYaml .Values.alertmanager.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 8 }} + {{- if .Values.alertmanager.podLabels}} + {{ toYaml .Values.alertmanager.podLabels | nindent 8 }} + {{- end}} + spec: +{{- if .Values.alertmanager.schedulerName }} + schedulerName: "{{ .Values.alertmanager.schedulerName }}" +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.alertmanager" . }} + {{- if .Values.alertmanager.extraInitContainers }} + initContainers: +{{ toYaml .Values.alertmanager.extraInitContainers | indent 8 }} + {{- end }} +{{- if .Values.alertmanager.priorityClassName }} + priorityClassName: "{{ .Values.alertmanager.priorityClassName }}" +{{- end }} + containers: + - name: {{ template "prometheus.name" . }}-{{ .Values.alertmanager.name }} + image: "{{ .Values.alertmanager.image.registry }}/{{ .Values.alertmanager.image.repository }}:{{ .Values.alertmanager.image.tag }}" + imagePullPolicy: "{{ .Values.alertmanager.image.pullPolicy }}" + env: + {{- range $key, $value := .Values.alertmanager.extraEnv }} + - name: {{ $key }} + value: {{ $value }} + {{- end }} + - name: POD_IP + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: status.podIP + args: + - --config.file=/etc/config/{{ .Values.alertmanager.configFileName }} + - --storage.path={{ .Values.alertmanager.persistentVolume.mountPath }} + {{- if .Values.alertmanager.service.enableMeshPeer }} + - --cluster.listen-address=0.0.0.0:6783 + - --cluster.advertise-address=[$(POD_IP)]:6783 + {{- else }} + - --cluster.listen-address= + {{- end }} + {{- range $key, $value := .Values.alertmanager.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.alertmanager.baseURL }} + - --web.external-url={{ .Values.alertmanager.baseURL }} + {{- end }} + {{- range .Values.alertmanager.clusterPeers }} + - --cluster.peer={{ . }} + {{- end }} + + ports: + - containerPort: 9093 + readinessProbe: + httpGet: + path: {{ .Values.alertmanager.prefixURL }}/-/ready + port: 9093 + {{- if .Values.alertmanager.probeHeaders }} + httpHeaders: + {{- range .Values.alertmanager.probeHeaders }} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + initialDelaySeconds: 30 + timeoutSeconds: 30 + resources: +{{ toYaml .Values.alertmanager.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: storage-volume + mountPath: "{{ .Values.alertmanager.persistentVolume.mountPath }}" + subPath: "{{ .Values.alertmanager.persistentVolume.subPath }}" + {{- range .Values.alertmanager.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.alertmanager.extraConfigmapMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + + {{- if .Values.configmapReload.alertmanager.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.alertmanager.name }}-{{ .Values.configmapReload.alertmanager.name }} + image: "{{ .Values.configmapReload.alertmanager.image.registry }}/{{ .Values.configmapReload.alertmanager.image.repository }}:{{ .Values.configmapReload.alertmanager.image.tag }}" + imagePullPolicy: "{{ .Values.configmapReload.alertmanager.image.pullPolicy }}" + args: + - --volume-dir=/etc/config + - --webhook-url=http://127.0.0.1:9093{{ .Values.alertmanager.prefixURL }}/-/reload + {{- range $key, $value := .Values.configmapReload.alertmanager.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- range .Values.configmapReload.alertmanager.extraVolumeDirs }} + - --volume-dir={{ . }} + {{- end }} + {{- if .Values.configmapReload.alertmanager.containerPort }} + ports: + - containerPort: {{ .Values.configmapReload.alertmanager.containerPort }} + {{- end }} + resources: +{{ toYaml .Values.configmapReload.alertmanager.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- range .Values.configmapReload.alertmanager.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.alertmanager.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.nodeSelector }} + nodeSelector: +{{ toYaml .Values.alertmanager.nodeSelector | indent 8 }} + {{- end }} + {{- with .Values.alertmanager.dnsConfig }} + dnsConfig: +{{ toYaml . | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.securityContext }} + securityContext: +{{ toYaml .Values.alertmanager.securityContext | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.tolerations }} + tolerations: +{{ toYaml .Values.alertmanager.tolerations | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.affinity }} + affinity: +{{ toYaml .Values.alertmanager.affinity | indent 8 }} + {{- end }} + volumes: + - name: config-volume + {{- if empty .Values.alertmanager.configFromSecret }} + configMap: + name: {{ if .Values.alertmanager.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.alertmanager.configMapOverrideName }}{{- else }}{{ template "prometheus.alertmanager.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.alertmanager.configFromSecret }} + {{- end }} + {{- range .Values.alertmanager.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- range .Values.alertmanager.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.configmapReload.alertmanager.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.alertmanager.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + - name: storage-volume + {{- if .Values.alertmanager.persistentVolume.enabled }} + persistentVolumeClaim: + claimName: {{ if .Values.alertmanager.persistentVolume.existingClaim }}{{ .Values.alertmanager.persistentVolume.existingClaim }}{{- else }}{{ template "prometheus.alertmanager.fullname" . }}{{- end }} + {{- else }} + emptyDir: + {{- if .Values.alertmanager.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.alertmanager.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} + {{- end -}} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/headless-svc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/headless-svc.yaml new file mode 100644 index 000000000..8c402c408 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/headless-svc.yaml @@ -0,0 +1,31 @@ +{{- if and .Values.alertmanager.enabled .Values.alertmanager.statefulSet.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.alertmanager.statefulSet.headless.annotations }} + annotations: +{{ toYaml .Values.alertmanager.statefulSet.headless.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} +{{- if .Values.alertmanager.statefulSet.headless.labels }} +{{ toYaml .Values.alertmanager.statefulSet.headless.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.alertmanager.fullname" . }}-headless +{{ include "prometheus.namespace" . | indent 2 }} +spec: + clusterIP: None + ports: + - name: http + port: {{ .Values.alertmanager.statefulSet.headless.servicePort }} + protocol: TCP + targetPort: 9093 +{{- if .Values.alertmanager.statefulSet.headless.enableMeshPeer }} + - name: meshpeer + port: 6783 + protocol: TCP + targetPort: 6783 +{{- end }} + selector: + {{- include "prometheus.alertmanager.matchLabels" . | nindent 4 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/ingress.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/ingress.yaml new file mode 100644 index 000000000..2a7b67c08 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/ingress.yaml @@ -0,0 +1,57 @@ +{{- if and .Values.alertmanager.enabled .Values.alertmanager.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "ingress.supportsPathType" .) "true" -}} +{{- $releaseName := .Release.Name -}} +{{- $serviceName := include "prometheus.alertmanager.fullname" . }} +{{- $servicePort := .Values.alertmanager.service.servicePort -}} +{{- $ingressPath := .Values.alertmanager.ingress.path -}} +{{- $ingressPathType := .Values.alertmanager.ingress.pathType -}} +{{- $extraPaths := .Values.alertmanager.ingress.extraPaths -}} +apiVersion: {{ template "ingress.apiVersion" . }} +kind: Ingress +metadata: +{{- if .Values.alertmanager.ingress.annotations }} + annotations: +{{ toYaml .Values.alertmanager.ingress.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} +{{- range $key, $value := .Values.alertmanager.ingress.extraLabels }} + {{ $key }}: {{ $value }} +{{- end }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.alertmanager.ingress.ingressClassName }} + ingressClassName: {{ .Values.alertmanager.ingress.ingressClassName }} + {{- end }} + rules: + {{- range .Values.alertmanager.ingress.hosts }} + {{- $url := splitList "/" . }} + - host: {{ first $url }} + http: + paths: +{{ if $extraPaths }} +{{ toYaml $extraPaths | indent 10 }} +{{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end -}} +{{- if .Values.alertmanager.ingress.tls }} + tls: +{{ toYaml .Values.alertmanager.ingress.tls | indent 4 }} + {{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/netpol.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/netpol.yaml new file mode 100644 index 000000000..e44ade60e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/netpol.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.alertmanager.enabled .Values.networkPolicy.enabled -}} +apiVersion: {{ template "prometheus.networkPolicy.apiVersion" . }} +kind: NetworkPolicy +metadata: + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "prometheus.alertmanager.matchLabels" . | nindent 6 }} + ingress: + - from: + - podSelector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 12 }} + - ports: + - port: 9093 +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pdb.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pdb.yaml new file mode 100644 index 000000000..41a92f364 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.alertmanager.podDisruptionBudget.enabled }} +apiVersion: policy/v1beta1 +kind: PodDisruptionBudget +metadata: + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.alertmanager.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "prometheus.alertmanager.labels" . | nindent 6 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/psp.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/psp.yaml new file mode 100644 index 000000000..64fb13003 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/psp.yaml @@ -0,0 +1,46 @@ +{{- if and .Values.alertmanager.enabled .Values.rbac.create .Values.podSecurityPolicy.enabled }} +apiVersion: {{ template "prometheus.podSecurityPolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "prometheus.alertmanager.fullname" . }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + annotations: +{{- if .Values.alertmanager.podSecurityPolicy.annotations }} +{{ toYaml .Values.alertmanager.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'persistentVolumeClaim' + - 'emptyDir' + - 'secret' + allowedHostPaths: + - pathPrefix: /etc + readOnly: true + - pathPrefix: {{ .Values.alertmanager.persistentVolume.mountPath }} + hostNetwork: false + hostPID: false + hostIPC: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: true +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pvc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pvc.yaml new file mode 100644 index 000000000..160e296a5 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/pvc.yaml @@ -0,0 +1,37 @@ +{{- if not .Values.alertmanager.statefulSet.enabled -}} +{{- if and .Values.alertmanager.enabled .Values.alertmanager.persistentVolume.enabled -}} +{{- if not .Values.alertmanager.persistentVolume.existingClaim -}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + {{- if .Values.alertmanager.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.alertmanager.persistentVolume.annotations | indent 4 }} + {{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + accessModes: +{{ toYaml .Values.alertmanager.persistentVolume.accessModes | indent 4 }} +{{- if .Values.alertmanager.persistentVolume.storageClass }} +{{- if (eq "-" .Values.alertmanager.persistentVolume.storageClass) }} + storageClassName: "" +{{- else }} + storageClassName: "{{ .Values.alertmanager.persistentVolume.storageClass }}" +{{- end }} +{{- end }} +{{- if .Values.alertmanager.persistentVolume.volumeBindingMode }} + volumeBindingMode: "{{ .Values.alertmanager.persistentVolume.volumeBindingMode }}" +{{- end }} + resources: + requests: + storage: "{{ .Values.alertmanager.persistentVolume.size }}" +{{- if .Values.alertmanager.persistentVolume.selector }} + selector: + {{- toYaml .Values.alertmanager.persistentVolume.selector | nindent 4 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/role.yaml new file mode 100644 index 000000000..ce60eaf0a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/role.yaml @@ -0,0 +1,24 @@ +{{- if and .Values.alertmanager.enabled .Values.rbac.create (eq .Values.alertmanager.useClusterRole false) (not .Values.alertmanager.useExistingRole) -}} +{{- range $.Values.alertmanager.namespaces }} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: Role +metadata: + labels: + {{- include "prometheus.alertmanager.labels" $ | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" $ }} + namespace: {{ . }} +rules: +{{- if $.Values.podSecurityPolicy.enabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ template "prometheus.alertmanager.fullname" $ }} +{{- else }} + [] +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/rolebinding.yaml new file mode 100644 index 000000000..906d6522d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/rolebinding.yaml @@ -0,0 +1,23 @@ +{{- if and .Values.alertmanager.enabled .Values.rbac.create (eq .Values.alertmanager.useClusterRole false) -}} +{{ range $.Values.alertmanager.namespaces }} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: RoleBinding +metadata: + labels: + {{- include "prometheus.alertmanager.labels" $ | nindent 4 }} + name: {{ template "prometheus.alertmanager.fullname" $ }} + namespace: {{ . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.alertmanager" $ }} +{{ include "prometheus.namespace" $ | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role +{{- if (not $.Values.alertmanager.useExistingRole) }} + name: {{ template "prometheus.alertmanager.fullname" $ }} +{{- else }} + name: {{ $.Values.alertmanager.useExistingRole }} +{{- end }} +{{- end }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/service.yaml new file mode 100644 index 000000000..9edc9ac65 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/service.yaml @@ -0,0 +1,53 @@ +{{- if .Values.alertmanager.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.alertmanager.service.annotations }} + annotations: +{{ toYaml .Values.alertmanager.service.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} +{{- if .Values.alertmanager.service.labels }} +{{ toYaml .Values.alertmanager.service.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: +{{- if .Values.alertmanager.service.clusterIP }} + clusterIP: {{ .Values.alertmanager.service.clusterIP }} +{{- end }} +{{- if .Values.alertmanager.service.externalIPs }} + externalIPs: +{{ toYaml .Values.alertmanager.service.externalIPs | indent 4 }} +{{- end }} +{{- if .Values.alertmanager.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.alertmanager.service.loadBalancerIP }} +{{- end }} +{{- if .Values.alertmanager.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.alertmanager.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - name: http + port: {{ .Values.alertmanager.service.servicePort }} + protocol: TCP + targetPort: 9093 + {{- if .Values.alertmanager.service.nodePort }} + nodePort: {{ .Values.alertmanager.service.nodePort }} + {{- end }} +{{- if .Values.alertmanager.service.enableMeshPeer }} + - name: meshpeer + port: 6783 + protocol: TCP + targetPort: 6783 +{{- end }} + selector: + {{- include "prometheus.alertmanager.matchLabels" . | nindent 4 }} +{{- if .Values.alertmanager.service.sessionAffinity }} + sessionAffinity: {{ .Values.alertmanager.service.sessionAffinity }} +{{- end }} + type: "{{ .Values.alertmanager.service.type }}" +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/serviceaccount.yaml new file mode 100644 index 000000000..a5d996a85 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/serviceaccount.yaml @@ -0,0 +1,11 @@ +{{- if and .Values.alertmanager.enabled .Values.serviceAccounts.alertmanager.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + name: {{ template "prometheus.serviceAccountName.alertmanager" . }} +{{ include "prometheus.namespace" . | indent 2 }} + annotations: +{{ toYaml .Values.serviceAccounts.alertmanager.annotations | indent 4 }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/sts.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/sts.yaml new file mode 100644 index 000000000..b978108ac --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/alertmanager/sts.yaml @@ -0,0 +1,188 @@ +{{- if and .Values.alertmanager.enabled .Values.alertmanager.statefulSet.enabled -}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: +{{- if .Values.alertmanager.statefulSet.annotations }} + annotations: + {{ toYaml .Values.alertmanager.statefulSet.annotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 4 }} + {{- if .Values.alertmanager.statefulSet.labels}} + {{ toYaml .Values.alertmanager.statefulSet.labels | nindent 4 }} + {{- end}} + name: {{ template "prometheus.alertmanager.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + serviceName: {{ template "prometheus.alertmanager.fullname" . }}-headless + selector: + matchLabels: + {{- include "prometheus.alertmanager.matchLabels" . | nindent 6 }} + replicas: {{ .Values.alertmanager.replicaCount }} + podManagementPolicy: {{ .Values.alertmanager.statefulSet.podManagementPolicy }} + template: + metadata: + {{- if .Values.alertmanager.podAnnotations }} + annotations: + {{ toYaml .Values.alertmanager.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.alertmanager.labels" . | nindent 8 }} + {{- if .Values.alertmanager.podLabels}} + {{ toYaml .Values.alertmanager.podLabels | nindent 8 }} + {{- end}} + spec: +{{- if .Values.alertmanager.affinity }} + affinity: +{{ toYaml .Values.alertmanager.affinity | indent 8 }} +{{- end }} +{{- if .Values.alertmanager.schedulerName }} + schedulerName: "{{ .Values.alertmanager.schedulerName }}" +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.alertmanager" . }} +{{- if .Values.alertmanager.priorityClassName }} + priorityClassName: "{{ .Values.alertmanager.priorityClassName }}" +{{- end }} + containers: + - name: {{ template "prometheus.name" . }}-{{ .Values.alertmanager.name }} + image: "{{ .Values.alertmanager.image.registry }}/{{ .Values.alertmanager.image.repository }}:{{ .Values.alertmanager.image.tag }}" + imagePullPolicy: "{{ .Values.alertmanager.image.pullPolicy }}" + env: + {{- range $key, $value := .Values.alertmanager.extraEnv }} + - name: {{ $key }} + value: {{ $value }} + {{- end }} + - name: POD_IP + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: status.podIP + args: + - --config.file=/etc/config/alertmanager.yml + - --storage.path={{ .Values.alertmanager.persistentVolume.mountPath }} + {{- if .Values.alertmanager.statefulSet.headless.enableMeshPeer }} + - --cluster.advertise-address=[$(POD_IP)]:6783 + - --cluster.listen-address=0.0.0.0:6783 + {{- range $n := until (.Values.alertmanager.replicaCount | int) }} + - --cluster.peer={{ template "prometheus.alertmanager.fullname" $ }}-{{ $n }}.{{ template "prometheus.alertmanager.fullname" $ }}-headless:6783 + {{- end }} + {{- else }} + - --cluster.listen-address= + {{- end }} + {{- range $key, $value := .Values.alertmanager.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.alertmanager.baseURL }} + - --web.external-url={{ .Values.alertmanager.baseURL }} + {{- end }} + + ports: + - containerPort: 9093 + {{- if .Values.alertmanager.statefulSet.headless.enableMeshPeer }} + - containerPort: 6783 + {{- end }} + readinessProbe: + httpGet: + path: {{ .Values.alertmanager.prefixURL }}/#/status + port: 9093 + initialDelaySeconds: 30 + timeoutSeconds: 30 + resources: +{{ toYaml .Values.alertmanager.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: storage-volume + mountPath: "{{ .Values.alertmanager.persistentVolume.mountPath }}" + subPath: "{{ .Values.alertmanager.persistentVolume.subPath }}" + {{- range .Values.alertmanager.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.configmapReload.alertmanager.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.alertmanager.name }}-{{ .Values.configmapReload.alertmanager.name }} + image: "{{ .Values.configmapReload.alertmanager.image.registry }}/{{ .Values.configmapReload.alertmanager.image.repository }}:{{ .Values.configmapReload.alertmanager.image.tag }}" + imagePullPolicy: "{{ .Values.configmapReload.alertmanager.image.pullPolicy }}" + args: + - --volume-dir=/etc/config + - --webhook-url=http://localhost:9093{{ .Values.alertmanager.prefixURL }}/-/reload + {{- range $key, $value := .Values.configmapReload.alertmanager.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.configmapReload.alertmanager.port }} + ports: + - containerPort: {{ .Values.configmapReload.alertmanager.port }} + {{- end }} + resources: +{{ toYaml .Values.configmapReload.alertmanager.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.nodeSelector }} + nodeSelector: +{{ toYaml .Values.alertmanager.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.securityContext }} + securityContext: +{{ toYaml .Values.alertmanager.securityContext | indent 8 }} + {{- end }} + {{- if .Values.alertmanager.tolerations }} + tolerations: +{{ toYaml .Values.alertmanager.tolerations | indent 8 }} + {{- end }} + volumes: + - name: config-volume + {{- if empty .Values.alertmanager.configFromSecret }} + configMap: + name: {{ if .Values.alertmanager.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.alertmanager.configMapOverrideName }}{{- else }}{{ template "prometheus.alertmanager.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.alertmanager.configFromSecret }} + {{- end }} + {{- range .Values.alertmanager.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} +{{- if .Values.alertmanager.persistentVolume.enabled }} + volumeClaimTemplates: + - metadata: + name: storage-volume + {{- if .Values.alertmanager.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.alertmanager.persistentVolume.annotations | indent 10 }} + {{- end }} + spec: + accessModes: +{{ toYaml .Values.alertmanager.persistentVolume.accessModes | indent 10 }} + resources: + requests: + storage: "{{ .Values.alertmanager.persistentVolume.size }}" + {{- if .Values.server.persistentVolume.storageClass }} + {{- if (eq "-" .Values.server.persistentVolume.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.alertmanager.persistentVolume.storageClass }}" + {{- end }} + {{- end }} +{{- else }} + - name: storage-volume + emptyDir: + {{- if .Values.alertmanager.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.alertmanager.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/daemonset.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/daemonset.yaml new file mode 100644 index 000000000..010f790a9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/daemonset.yaml @@ -0,0 +1,150 @@ +{{- if .Values.nodeExporter.enabled -}} +apiVersion: {{ template "prometheus.daemonset.apiVersion" . }} +kind: DaemonSet +metadata: +{{- if .Values.nodeExporter.deploymentAnnotations }} + annotations: +{{ toYaml .Values.nodeExporter.deploymentAnnotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} + name: {{ template "prometheus.nodeExporter.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + selector: + matchLabels: + {{- include "prometheus.nodeExporter.matchLabels" . | nindent 6 }} + {{- if .Values.nodeExporter.updateStrategy }} + updateStrategy: +{{ toYaml .Values.nodeExporter.updateStrategy | indent 4 }} + {{- end }} + template: + metadata: + {{- if .Values.nodeExporter.podAnnotations }} + annotations: +{{ toYaml .Values.nodeExporter.podAnnotations | indent 8 }} + {{- end }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 8 }} +{{- if .Values.nodeExporter.pod.labels }} +{{ toYaml .Values.nodeExporter.pod.labels | indent 8 }} +{{- end }} + spec: + serviceAccountName: {{ template "prometheus.serviceAccountName.nodeExporter" . }} + {{- if .Values.nodeExporter.extraInitContainers }} + initContainers: +{{ toYaml .Values.nodeExporter.extraInitContainers | indent 8 }} + {{- end }} +{{- if .Values.nodeExporter.priorityClassName }} + priorityClassName: "{{ .Values.nodeExporter.priorityClassName }}" +{{- end }} + containers: + - name: {{ template "prometheus.name" . }}-{{ .Values.nodeExporter.name }} + image: "{{ .Values.nodeExporter.image.registry }}/{{ .Values.nodeExporter.image.repository }}:{{ .Values.nodeExporter.image.tag }}" + imagePullPolicy: "{{ .Values.nodeExporter.image.pullPolicy }}" + args: + - --path.procfs=/host/proc + - --path.sysfs=/host/sys + {{- if .Values.nodeExporter.hostRootfs }} + - --path.rootfs=/host/root + {{- end }} + {{- if .Values.nodeExporter.hostNetwork }} + - --web.listen-address=:{{ .Values.nodeExporter.service.hostPort }} + {{- end }} + {{- range $key, $value := .Values.nodeExporter.extraArgs }} + {{- if $value }} + - --{{ $key }}={{ $value }} + {{- else }} + - --{{ $key }} + {{- end }} + {{- end }} + ports: + - name: metrics + {{- if .Values.nodeExporter.hostNetwork }} + containerPort: {{ .Values.nodeExporter.service.hostPort }} + {{- else }} + containerPort: 9100 + {{- end }} + hostPort: {{ .Values.nodeExporter.service.hostPort }} + resources: +{{ toYaml .Values.nodeExporter.resources | indent 12 }} + {{- if .Values.nodeExporter.container.securityContext }} + securityContext: +{{ toYaml .Values.nodeExporter.container.securityContext | indent 12 }} + {{- end }} + volumeMounts: + - name: proc + mountPath: /host/proc + readOnly: true + - name: sys + mountPath: /host/sys + readOnly: true + {{- if .Values.nodeExporter.hostRootfs }} + - name: root + mountPath: /host/root + mountPropagation: HostToContainer + readOnly: true + {{- end }} + {{- range .Values.nodeExporter.extraHostPathMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + {{- if .mountPropagation }} + mountPropagation: {{ .mountPropagation }} + {{- end }} + {{- end }} + {{- range .Values.nodeExporter.extraConfigmapMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.nodeExporter.hostNetwork }} + hostNetwork: true + {{- end }} + {{- if .Values.nodeExporter.hostPID }} + hostPID: true + {{- end }} + {{- if .Values.nodeExporter.tolerations }} + tolerations: +{{ toYaml .Values.nodeExporter.tolerations | indent 8 }} + {{- end }} + {{- if .Values.nodeExporter.nodeSelector }} + nodeSelector: +{{ toYaml .Values.nodeExporter.nodeSelector | indent 8 }} + {{- end }} + {{- with .Values.nodeExporter.dnsConfig }} + dnsConfig: +{{ toYaml . | indent 8 }} + {{- end }} + {{- if .Values.nodeExporter.securityContext }} + securityContext: +{{ toYaml .Values.nodeExporter.securityContext | indent 8 }} + {{- end }} + volumes: + - name: proc + hostPath: + path: /proc + - name: sys + hostPath: + path: /sys + {{- if .Values.nodeExporter.hostRootfs }} + - name: root + hostPath: + path: / + {{- end }} + {{- range .Values.nodeExporter.extraHostPathMounts }} + - name: {{ .name }} + hostPath: + path: {{ .hostPath }} + {{- end }} + {{- range .Values.nodeExporter.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/psp.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/psp.yaml new file mode 100644 index 000000000..bd9c73bee --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/psp.yaml @@ -0,0 +1,55 @@ +{{- if and .Values.nodeExporter.enabled .Values.rbac.create .Values.podSecurityPolicy.enabled }} +apiVersion: {{ template "prometheus.podSecurityPolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "prometheus.nodeExporter.fullname" . }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} + annotations: +{{- if .Values.nodeExporter.podSecurityPolicy.annotations }} +{{ toYaml .Values.nodeExporter.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'configMap' + - 'hostPath' + - 'secret' + allowedHostPaths: + - pathPrefix: /proc + readOnly: true + - pathPrefix: /sys + readOnly: true + - pathPrefix: / + readOnly: true + {{- range .Values.nodeExporter.extraHostPathMounts }} + - pathPrefix: {{ .hostPath }} + readOnly: {{ .readOnly }} + {{- end }} + hostNetwork: {{ .Values.nodeExporter.hostNetwork }} + hostPID: {{ .Values.nodeExporter.hostPID }} + hostIPC: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false + hostPorts: + - min: 1 + max: 65535 +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/role.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/role.yaml new file mode 100644 index 000000000..d8ef3ed90 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/role.yaml @@ -0,0 +1,17 @@ +{{- if and .Values.nodeExporter.enabled .Values.rbac.create }} +{{- if or (default .Values.nodeExporter.podSecurityPolicy.enabled false) (.Values.podSecurityPolicy.enabled) }} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: Role +metadata: + name: {{ template "prometheus.nodeExporter.fullname" . }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} +{{ include "prometheus.namespace" . | indent 2 }} +rules: +- apiGroups: ['extensions'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ template "prometheus.nodeExporter.fullname" . }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/rolebinding.yaml new file mode 100644 index 000000000..06914b70a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/rolebinding.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.nodeExporter.enabled .Values.rbac.create }} +{{- if .Values.podSecurityPolicy.enabled }} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: RoleBinding +metadata: + name: {{ template "prometheus.nodeExporter.fullname" . }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} +{{ include "prometheus.namespace" . | indent 2 }} +roleRef: + kind: Role + name: {{ template "prometheus.nodeExporter.fullname" . }} + apiGroup: rbac.authorization.k8s.io +subjects: +- kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.nodeExporter" . }} +{{ include "prometheus.namespace" . | indent 2 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/serviceaccount.yaml new file mode 100644 index 000000000..0cf91afba --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/serviceaccount.yaml @@ -0,0 +1,11 @@ +{{- if and .Values.nodeExporter.enabled .Values.serviceAccounts.nodeExporter.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} + name: {{ template "prometheus.serviceAccountName.nodeExporter" . }} +{{ include "prometheus.namespace" . | indent 2 }} + annotations: +{{ toYaml .Values.serviceAccounts.nodeExporter.annotations | indent 4 }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/svc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/svc.yaml new file mode 100644 index 000000000..26d1eaa21 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/node-exporter/svc.yaml @@ -0,0 +1,47 @@ +{{- if .Values.nodeExporter.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.nodeExporter.service.annotations }} + annotations: +{{ toYaml .Values.nodeExporter.service.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.nodeExporter.labels" . | nindent 4 }} +{{- if .Values.nodeExporter.service.labels }} +{{ toYaml .Values.nodeExporter.service.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.nodeExporter.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: +{{- if .Values.nodeExporter.service.clusterIP }} + clusterIP: {{ .Values.nodeExporter.service.clusterIP }} +{{- end }} +{{- if .Values.nodeExporter.service.externalIPs }} + externalIPs: +{{ toYaml .Values.nodeExporter.service.externalIPs | indent 4 }} +{{- end }} +{{- if .Values.nodeExporter.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.nodeExporter.service.loadBalancerIP }} +{{- end }} +{{- if .Values.nodeExporter.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.nodeExporter.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - name: metrics + {{- if .Values.nodeExporter.hostNetwork }} + port: {{ .Values.nodeExporter.service.hostPort }} + protocol: TCP + targetPort: {{ .Values.nodeExporter.service.hostPort }} + {{- else }} + port: {{ .Values.nodeExporter.service.servicePort }} + protocol: TCP + targetPort: 9100 + {{- end }} + selector: + {{- include "prometheus.nodeExporter.matchLabels" . | nindent 4 }} + type: "{{ .Values.nodeExporter.service.type }}" +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrole.yaml new file mode 100644 index 000000000..76ecf053f --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrole.yaml @@ -0,0 +1,21 @@ +{{- if and .Values.pushgateway.enabled .Values.rbac.create -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRole +metadata: + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }} +rules: +{{- if .Values.podSecurityPolicy.enabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ template "prometheus.pushgateway.fullname" . }} +{{- else }} + [] +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrolebinding.yaml new file mode 100644 index 000000000..15770ee50 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.pushgateway.enabled .Values.rbac.create -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRoleBinding +metadata: + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.pushgateway" . }} +{{ include "prometheus.namespace" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ template "prometheus.pushgateway.fullname" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/deploy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/deploy.yaml new file mode 100644 index 000000000..9314a1f45 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/deploy.yaml @@ -0,0 +1,119 @@ +{{- if .Values.pushgateway.enabled -}} +apiVersion: {{ template "prometheus.deployment.apiVersion" . }} +kind: Deployment +metadata: +{{- if .Values.pushgateway.deploymentAnnotations }} + annotations: + {{ toYaml .Values.pushgateway.deploymentAnnotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + selector: + {{- if .Values.schedulerName }} + schedulerName: "{{ .Values.schedulerName }}" + {{- end }} + matchLabels: + {{- include "prometheus.pushgateway.matchLabels" . | nindent 6 }} + replicas: {{ .Values.pushgateway.replicaCount }} + {{- if .Values.pushgateway.strategy }} + strategy: +{{ toYaml .Values.pushgateway.strategy | trim | indent 4 }} + {{ if eq .Values.pushgateway.strategy.type "Recreate" }}rollingUpdate: null{{ end }} +{{- end }} + template: + metadata: + {{- if .Values.pushgateway.podAnnotations }} + annotations: + {{ toYaml .Values.pushgateway.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 8 }} + {{- if .Values.pushgateway.podLabels }} + {{ toYaml .Values.pushgateway.podLabels | nindent 8 }} + {{- end }} + spec: + serviceAccountName: {{ template "prometheus.serviceAccountName.pushgateway" . }} + {{- if .Values.pushgateway.extraInitContainers }} + initContainers: +{{ toYaml .Values.pushgateway.extraInitContainers | indent 8 }} + {{- end }} +{{- if .Values.pushgateway.priorityClassName }} + priorityClassName: "{{ .Values.pushgateway.priorityClassName }}" +{{- end }} + containers: + - name: {{ template "prometheus.name" . }}-{{ .Values.pushgateway.name }} + image: "{{ .Values.pushgateway.image.registry }}/{{ .Values.pushgateway.image.repository }}:{{ .Values.pushgateway.image.tag }}" + imagePullPolicy: "{{ .Values.pushgateway.image.pullPolicy }}" + args: + {{- range $key, $value := .Values.pushgateway.extraArgs }} + {{- $stringvalue := toString $value }} + {{- if eq $stringvalue "true" }} + - --{{ $key }} + {{- else }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- end }} + ports: + - containerPort: 9091 + livenessProbe: + httpGet: + {{- if (index .Values "pushgateway" "extraArgs" "web.route-prefix") }} + path: /{{ index .Values "pushgateway" "extraArgs" "web.route-prefix" }}/-/healthy + {{- else }} + path: /-/healthy + {{- end }} + port: 9091 + initialDelaySeconds: 10 + timeoutSeconds: 10 + readinessProbe: + httpGet: + {{- if (index .Values "pushgateway" "extraArgs" "web.route-prefix") }} + path: /{{ index .Values "pushgateway" "extraArgs" "web.route-prefix" }}/-/ready + {{- else }} + path: /-/ready + {{- end }} + port: 9091 + initialDelaySeconds: 10 + timeoutSeconds: 10 + resources: +{{ toYaml .Values.pushgateway.resources | indent 12 }} + {{- if .Values.pushgateway.persistentVolume.enabled }} + volumeMounts: + - name: storage-volume + mountPath: "{{ .Values.pushgateway.persistentVolume.mountPath }}" + subPath: "{{ .Values.pushgateway.persistentVolume.subPath }}" + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.pushgateway.nodeSelector }} + nodeSelector: +{{ toYaml .Values.pushgateway.nodeSelector | indent 8 }} + {{- end }} + {{- with .Values.pushgateway.dnsConfig }} + dnsConfig: +{{ toYaml . | indent 8 }} + {{- end }} + {{- if .Values.pushgateway.securityContext }} + securityContext: +{{ toYaml .Values.pushgateway.securityContext | indent 8 }} + {{- end }} + {{- if .Values.pushgateway.tolerations }} + tolerations: +{{ toYaml .Values.pushgateway.tolerations | indent 8 }} + {{- end }} + {{- if .Values.pushgateway.affinity }} + affinity: +{{ toYaml .Values.pushgateway.affinity | indent 8 }} + {{- end }} + {{- if .Values.pushgateway.persistentVolume.enabled }} + volumes: + - name: storage-volume + persistentVolumeClaim: + claimName: {{ if .Values.pushgateway.persistentVolume.existingClaim }}{{ .Values.pushgateway.persistentVolume.existingClaim }}{{- else }}{{ template "prometheus.pushgateway.fullname" . }}{{- end }} + {{- end -}} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/ingress.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/ingress.yaml new file mode 100644 index 000000000..2ff72abd5 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/ingress.yaml @@ -0,0 +1,54 @@ +{{- if and .Values.pushgateway.enabled .Values.pushgateway.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "ingress.supportsPathType" .) "true" -}} +{{- $releaseName := .Release.Name -}} +{{- $serviceName := include "prometheus.pushgateway.fullname" . }} +{{- $servicePort := .Values.pushgateway.service.servicePort -}} +{{- $ingressPath := .Values.pushgateway.ingress.path -}} +{{- $ingressPathType := .Values.pushgateway.ingress.pathType -}} +{{- $extraPaths := .Values.pushgateway.ingress.extraPaths -}} +apiVersion: {{ template "ingress.apiVersion" . }} +kind: Ingress +metadata: +{{- if .Values.pushgateway.ingress.annotations }} + annotations: +{{ toYaml .Values.pushgateway.ingress.annotations | indent 4}} +{{- end }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.pushgateway.ingress.ingressClassName }} + ingressClassName: {{ .Values.pushgateway.ingress.ingressClassName }} + {{- end }} + rules: + {{- range .Values.pushgateway.ingress.hosts }} + {{- $url := splitList "/" . }} + - host: {{ first $url }} + http: + paths: +{{ if $extraPaths }} +{{ toYaml $extraPaths | indent 10 }} +{{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end -}} +{{- if .Values.pushgateway.ingress.tls }} + tls: +{{ toYaml .Values.pushgateway.ingress.tls | indent 4 }} + {{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/netpol.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/netpol.yaml new file mode 100644 index 000000000..c8d1fb37e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/netpol.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.pushgateway.enabled .Values.networkPolicy.enabled -}} +apiVersion: {{ template "prometheus.networkPolicy.apiVersion" . }} +kind: NetworkPolicy +metadata: + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "prometheus.pushgateway.matchLabels" . | nindent 6 }} + ingress: + - from: + - podSelector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 12 }} + - ports: + - port: 9091 +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pdb.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pdb.yaml new file mode 100644 index 000000000..50beb486d --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.pushgateway.podDisruptionBudget.enabled }} +apiVersion: policy/v1beta1 +kind: PodDisruptionBudget +metadata: + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.pushgateway.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "prometheus.pushgateway.labels" . | nindent 6 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/psp.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/psp.yaml new file mode 100644 index 000000000..1ca3267f8 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/psp.yaml @@ -0,0 +1,42 @@ +{{- if and .Values.pushgateway.enabled .Values.rbac.create .Values.podSecurityPolicy.enabled }} +apiVersion: {{ template "prometheus.podSecurityPolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "prometheus.pushgateway.fullname" . }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + annotations: +{{- if .Values.pushgateway.podSecurityPolicy.annotations }} +{{ toYaml .Values.pushgateway.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + requiredDropCapabilities: + - ALL + volumes: + - 'persistentVolumeClaim' + - 'secret' + allowedHostPaths: + - pathPrefix: {{ .Values.pushgateway.persistentVolume.mountPath }} + hostNetwork: false + hostPID: false + hostIPC: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: true +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pvc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pvc.yaml new file mode 100644 index 000000000..d5d64ddcc --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/pvc.yaml @@ -0,0 +1,31 @@ +{{- if .Values.pushgateway.persistentVolume.enabled -}} +{{- if not .Values.pushgateway.persistentVolume.existingClaim -}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + {{- if .Values.pushgateway.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.pushgateway.persistentVolume.annotations | indent 4 }} + {{- end }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + accessModes: +{{ toYaml .Values.pushgateway.persistentVolume.accessModes | indent 4 }} +{{- if .Values.pushgateway.persistentVolume.storageClass }} +{{- if (eq "-" .Values.pushgateway.persistentVolume.storageClass) }} + storageClassName: "" +{{- else }} + storageClassName: "{{ .Values.pushgateway.persistentVolume.storageClass }}" +{{- end }} +{{- end }} +{{- if .Values.pushgateway.persistentVolume.volumeBindingMode }} + volumeBindingMode: "{{ .Values.pushgateway.persistentVolume.volumeBindingMode }}" +{{- end }} + resources: + requests: + storage: "{{ .Values.pushgateway.persistentVolume.size }}" +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/service.yaml new file mode 100644 index 000000000..f05f17c42 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/service.yaml @@ -0,0 +1,41 @@ +{{- if .Values.pushgateway.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.pushgateway.service.annotations }} + annotations: +{{ toYaml .Values.pushgateway.service.annotations | indent 4}} +{{- end }} + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} +{{- if .Values.pushgateway.service.labels }} +{{ toYaml .Values.pushgateway.service.labels | indent 4}} +{{- end }} + name: {{ template "prometheus.pushgateway.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: +{{- if .Values.pushgateway.service.clusterIP }} + clusterIP: {{ .Values.pushgateway.service.clusterIP }} +{{- end }} +{{- if .Values.pushgateway.service.externalIPs }} + externalIPs: +{{ toYaml .Values.pushgateway.service.externalIPs | indent 4 }} +{{- end }} +{{- if .Values.pushgateway.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.pushgateway.service.loadBalancerIP }} +{{- end }} +{{- if .Values.pushgateway.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.pushgateway.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - name: http + port: {{ .Values.pushgateway.service.servicePort }} + protocol: TCP + targetPort: 9091 + selector: + {{- include "prometheus.pushgateway.matchLabels" . | nindent 4 }} + type: "{{ .Values.pushgateway.service.type }}" +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/serviceaccount.yaml new file mode 100644 index 000000000..8c0b876f3 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/serviceaccount.yaml @@ -0,0 +1,11 @@ +{{- if and .Values.pushgateway.enabled .Values.serviceAccounts.pushgateway.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.serviceAccountName.pushgateway" . }} +{{ include "prometheus.namespace" . | indent 2 }} + annotations: +{{ toYaml .Values.serviceAccounts.pushgateway.annotations | indent 4 }} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/vpa.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/vpa.yaml new file mode 100644 index 000000000..0ac54f9fe --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/pushgateway/vpa.yaml @@ -0,0 +1,20 @@ +{{- if .Values.pushgateway.enabled -}} +{{- if .Values.pushgateway.verticalAutoscaler.enabled -}} +apiVersion: autoscaling.k8s.io/v1beta2 +kind: VerticalPodAutoscaler +metadata: + labels: + {{- include "prometheus.pushgateway.labels" . | nindent 4 }} + name: {{ template "prometheus.pushgateway.fullname" . }}-vpa +{{ include "prometheus.namespace" . | indent 2 }} +spec: + targetRef: + apiVersion: "apps/v1" + kind: Deployment + name: {{ template "prometheus.pushgateway.fullname" . }} + updatePolicy: + updateMode: {{ .Values.pushgateway.verticalAutoscaler.updateMode | default "Off" | quote }} + resourcePolicy: + containerPolicies: {{ .Values.pushgateway.verticalAutoscaler.containerPolicies | default list | toYaml | trim | nindent 4 }} +{{- end -}} {{/* if .Values.pushgateway.verticalAutoscaler.enabled */}} +{{- end -}} {{/* .Values.pushgateway.enabled */}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrole.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrole.yaml new file mode 100644 index 000000000..2520235ab --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrole.yaml @@ -0,0 +1,48 @@ +{{- if and .Values.server.enabled .Values.rbac.create (empty .Values.server.useExistingClusterRoleName) -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRole +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} +rules: +{{- if .Values.podSecurityPolicy.enabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + verbs: + - use + resourceNames: + - {{ template "prometheus.server.fullname" . }} +{{- end }} + - apiGroups: + - "" + resources: + - nodes + - nodes/proxy + - nodes/metrics + - services + - endpoints + - pods + - ingresses + - configmaps + verbs: + - get + - list + - watch + - apiGroups: + - "extensions" + - "networking.k8s.io" + resources: + - ingresses/status + - ingresses + verbs: + - get + - list + - watch + - nonResourceURLs: + - "/metrics" + verbs: + - get +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrolebinding.yaml new file mode 100644 index 000000000..5a79611ff --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and .Values.server.enabled .Values.rbac.create (empty .Values.server.namespaces) (empty .Values.server.useExistingClusterRoleName) -}} +apiVersion: {{ template "rbac.apiVersion" . }} +kind: ClusterRoleBinding +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.server" . }} +{{ include "prometheus.namespace" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ template "prometheus.server.fullname" . }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/cm.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/cm.yaml new file mode 100644 index 000000000..a0a813ae2 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/cm.yaml @@ -0,0 +1,85 @@ +{{- if .Values.server.enabled -}} +{{- if (empty .Values.server.configMapOverrideName) -}} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +data: +{{- $root := . -}} +{{- range $key, $value := .Values.ruleFiles }} + {{ $key }}: {{- toYaml $value | indent 2 }} +{{- end }} +{{- range $key, $value := .Values.serverFiles }} + {{ $key }}: | +{{- if eq $key "prometheus.yml" }} + global: +{{ $root.Values.server.global | toYaml | trimSuffix "\n" | indent 6 }} +{{- if $root.Values.server.remoteWrite }} + remote_write: +{{ $root.Values.server.remoteWrite | toYaml | indent 4 }} +{{- end }} +{{- if $root.Values.server.remoteRead }} + remote_read: +{{ $root.Values.server.remoteRead | toYaml | indent 4 }} +{{- end }} +{{- end }} +{{- if eq $key "alerts" }} +{{- if and (not (empty $value)) (empty $value.groups) }} + groups: +{{- range $ruleKey, $ruleValue := $value }} + - name: {{ $ruleKey -}}.rules + rules: +{{ $ruleValue | toYaml | trimSuffix "\n" | indent 6 }} +{{- end }} +{{- else }} +{{ toYaml $value | indent 4 }} +{{- end }} +{{- else }} +{{ toYaml $value | default "{}" | indent 4 }} +{{- end }} +{{- if eq $key "prometheus.yml" -}} +{{- if $root.Values.extraScrapeConfigs }} +{{ tpl $root.Values.extraScrapeConfigs $root | indent 4 }} +{{- end -}} +{{- if or ($root.Values.alertmanager.enabled) ($root.Values.server.alertmanagers) }} + alerting: +{{- if $root.Values.alertRelabelConfigs }} +{{ $root.Values.alertRelabelConfigs | toYaml | trimSuffix "\n" | indent 6 }} +{{- end }} + alertmanagers: +{{- if $root.Values.server.alertmanagers }} +{{ toYaml $root.Values.server.alertmanagers | indent 8 }} +{{- else }} + - kubernetes_sd_configs: + - role: pod + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + {{- if $root.Values.alertmanager.prefixURL }} + path_prefix: {{ $root.Values.alertmanager.prefixURL }} + {{- end }} + relabel_configs: + - source_labels: [__meta_kubernetes_namespace] + regex: {{ $root.Release.Namespace }} + action: keep + - source_labels: [__meta_kubernetes_pod_label_app] + regex: {{ template "prometheus.name" $root }} + action: keep + - source_labels: [__meta_kubernetes_pod_label_component] + regex: alertmanager + action: keep + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_probe] + regex: {{ index $root.Values.alertmanager.podAnnotations "prometheus.io/probe" | default ".*" }} + action: keep + - source_labels: [__meta_kubernetes_pod_container_port_number] + regex: "9093" + action: keep +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/deploy.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/deploy.yaml new file mode 100644 index 000000000..22808929e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/deploy.yaml @@ -0,0 +1,324 @@ +{{- if .Values.server.enabled -}} +{{- if not .Values.server.statefulSet.enabled -}} +apiVersion: {{ template "prometheus.deployment.apiVersion" . }} +kind: Deployment +metadata: +{{- if .Values.server.deploymentAnnotations }} + annotations: + {{ toYaml .Values.server.deploymentAnnotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + selector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + replicas: {{ .Values.server.replicaCount }} + {{- if .Values.server.strategy }} + strategy: +{{ toYaml .Values.server.strategy | trim | indent 4 }} + {{ if eq .Values.server.strategy.type "Recreate" }}rollingUpdate: null{{ end }} +{{- end }} + template: + metadata: + {{- if .Values.server.podAnnotations }} + annotations: + {{ toYaml .Values.server.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 8 }} + {{- if .Values.server.podLabels}} + {{ toYaml .Values.server.podLabels | nindent 8 }} + {{- end}} + spec: +{{- if .Values.server.priorityClassName }} + priorityClassName: "{{ .Values.server.priorityClassName }}" +{{- end }} +{{- if .Values.server.schedulerName }} + schedulerName: "{{ .Values.server.schedulerName }}" +{{- end }} +{{- if semverCompare ">=1.13-0" .Capabilities.KubeVersion.GitVersion }} + {{- if or (.Values.server.enableServiceLinks) (eq (.Values.server.enableServiceLinks | toString) "") }} + enableServiceLinks: true + {{- else }} + enableServiceLinks: false + {{- end }} +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.server" . }} + {{- if .Values.server.extraInitContainers }} + initContainers: +{{ toYaml .Values.server.extraInitContainers | indent 8 }} + {{- end }} + containers: + {{- if .Values.configmapReload.prometheus.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }}-{{ .Values.configmapReload.prometheus.name }} + image: "{{ .Values.configmapReload.prometheus.image.registry }}/{{ .Values.configmapReload.prometheus.image.repository }}:{{ .Values.configmapReload.prometheus.image.tag }}" + imagePullPolicy: "{{ .Values.configmapReload.prometheus.image.pullPolicy }}" + args: + - --volume-dir=/etc/config + - --webhook-url=http://127.0.0.1:9090{{ .Values.server.prefixURL }}/-/reload + {{- range $key, $value := .Values.configmapReload.prometheus.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraVolumeDirs }} + - --volume-dir={{ . }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + ports: + - containerPort: {{ .Values.configmapReload.prometheus.containerPort }} + {{- end }} + resources: +{{ toYaml .Values.configmapReload.prometheus.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- end }} + + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }} + image: "{{ .Values.server.image.registry }}/{{ .Values.server.image.repository }}:{{ .Values.server.image.tag }}" + imagePullPolicy: "{{ .Values.server.image.pullPolicy }}" + {{- if .Values.server.env }} + env: +{{ toYaml .Values.server.env | indent 12}} + {{- end }} + args: + {{- if .Values.server.defaultFlagsOverride }} + {{ toYaml .Values.server.defaultFlagsOverride | nindent 12}} + {{- else }} + {{- if .Values.server.retention }} + - --storage.tsdb.retention.time={{ .Values.server.retention }} + {{- end }} + - --config.file={{ .Values.server.configPath }} + {{- if .Values.server.storagePath }} + - --storage.tsdb.path={{ .Values.server.storagePath }} + {{- else }} + - --storage.tsdb.path={{ .Values.server.persistentVolume.mountPath }} + {{- end }} + - --web.console.libraries=/etc/prometheus/console_libraries + - --web.console.templates=/etc/prometheus/consoles + {{- range .Values.server.extraFlags }} + - --{{ . }} + {{- end }} + {{- range $key, $value := .Values.server.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.server.prefixURL }} + - --web.route-prefix={{ .Values.server.prefixURL }} + {{- end }} + {{- if .Values.server.baseURL }} + - --web.external-url={{ .Values.server.baseURL }} + {{- end }} + {{- end }} + ports: + - containerPort: 9090 + {{- if .Values.server.hostPort }} + hostPort: {{ .Values.server.hostPort }} + {{- end }} + readinessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/ready + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- if .Values.server.probeHeaders }} + httpHeaders: + {{- range .Values.server.probeHeaders}} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.readinessProbeInitialDelay }} + periodSeconds: {{ .Values.server.readinessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.readinessProbeTimeout }} + failureThreshold: {{ .Values.server.readinessProbeFailureThreshold }} + successThreshold: {{ .Values.server.readinessProbeSuccessThreshold }} + livenessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- if .Values.server.probeHeaders }} + httpHeaders: + {{- range .Values.server.probeHeaders}} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.livenessProbeInitialDelay }} + periodSeconds: {{ .Values.server.livenessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.livenessProbeTimeout }} + failureThreshold: {{ .Values.server.livenessProbeFailureThreshold }} + successThreshold: {{ .Values.server.livenessProbeSuccessThreshold }} + {{- if .Values.server.startupProbe.enabled }} + startupProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- if .Values.server.probeHeaders }} + httpHeaders: + {{- range .Values.server.probeHeaders}} + - name: {{ .name }} + value: {{ .value }} + {{- end }} + {{- end }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + failureThreshold: {{ .Values.server.startupProbe.failureThreshold }} + periodSeconds: {{ .Values.server.startupProbe.periodSeconds }} + timeoutSeconds: {{ .Values.server.startupProbe.timeoutSeconds }} + {{- end }} + resources: +{{ toYaml .Values.server.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: storage-volume + mountPath: {{ .Values.server.persistentVolume.mountPath }} + subPath: "{{ .Values.server.persistentVolume.subPath }}" + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.server.extraVolumeMounts }} + {{ toYaml .Values.server.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- if .Values.server.containerSecurityContext }} + securityContext: + {{- toYaml .Values.server.containerSecurityContext | nindent 12 }} + {{- end }} + {{- if .Values.server.sidecarContainers }} + {{- range $name, $spec := .Values.server.sidecarContainers }} + - name: {{ $name }} + {{- if kindIs "string" $spec }} + {{- tpl $spec $ | nindent 10 }} + {{- else }} + {{- toYaml $spec | nindent 10 }} + {{- end }} + {{- end }} + {{- end }} + hostNetwork: {{ .Values.server.hostNetwork }} + {{- if .Values.server.dnsPolicy }} + dnsPolicy: {{ .Values.server.dnsPolicy }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.server.nodeSelector }} + nodeSelector: +{{ toYaml .Values.server.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.server.hostAliases }} + hostAliases: +{{ toYaml .Values.server.hostAliases | indent 8 }} + {{- end }} + {{- if .Values.server.dnsConfig }} + dnsConfig: +{{ toYaml .Values.server.dnsConfig | indent 8 }} + {{- end }} + {{- if .Values.server.securityContext }} + securityContext: +{{ toYaml .Values.server.securityContext | indent 8 }} + {{- end }} + {{- if .Values.server.tolerations }} + tolerations: +{{ toYaml .Values.server.tolerations | indent 8 }} + {{- end }} + {{- if .Values.server.affinity }} + affinity: +{{ toYaml .Values.server.affinity | indent 8 }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.server.terminationGracePeriodSeconds }} + volumes: + - name: config-volume + {{- if empty .Values.server.configFromSecret }} + configMap: + name: {{ if .Values.server.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.server.configMapOverrideName }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.server.configFromSecret }} + {{- end }} + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + hostPath: + path: {{ .hostPath }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} +{{- if .Values.server.extraVolumes }} +{{ toYaml .Values.server.extraVolumes | indent 8}} +{{- end }} + - name: storage-volume + {{- if .Values.server.persistentVolume.enabled }} + persistentVolumeClaim: + claimName: {{ if .Values.server.persistentVolume.existingClaim }}{{ .Values.server.persistentVolume.existingClaim }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + emptyDir: + {{- if .Values.server.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.server.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/extra-manifests.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/extra-manifests.yaml new file mode 100644 index 000000000..e46d4d8b9 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.server.extraObjects }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/headless-svc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/headless-svc.yaml new file mode 100644 index 000000000..d519f4e0e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/headless-svc.yaml @@ -0,0 +1,37 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.server.statefulSet.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.server.statefulSet.headless.annotations }} + annotations: +{{ toYaml .Values.server.statefulSet.headless.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- if .Values.server.statefulSet.headless.labels }} +{{ toYaml .Values.server.statefulSet.headless.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }}-headless +{{ include "prometheus.namespace" . | indent 2 }} +spec: + clusterIP: None + ports: + - name: http + port: {{ .Values.server.statefulSet.headless.servicePort }} + protocol: TCP + targetPort: 9090 + {{- if .Values.server.statefulSet.headless.gRPC.enabled }} + - name: grpc + port: {{ .Values.server.statefulSet.headless.gRPC.servicePort }} + protocol: TCP + targetPort: 10901 + {{- if .Values.server.statefulSet.headless.gRPC.nodePort }} + nodePort: {{ .Values.server.statefulSet.headless.gRPC.nodePort }} + {{- end }} + {{- end }} + + selector: + {{- include "prometheus.server.matchLabels" . | nindent 4 }} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/ingress.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/ingress.yaml new file mode 100644 index 000000000..000f39cab --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/ingress.yaml @@ -0,0 +1,59 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.server.ingress.enabled -}} +{{- $ingressApiIsStable := eq (include "ingress.isStable" .) "true" -}} +{{- $ingressSupportsIngressClassName := eq (include "ingress.supportsIngressClassName" .) "true" -}} +{{- $ingressSupportsPathType := eq (include "ingress.supportsPathType" .) "true" -}} +{{- $releaseName := .Release.Name -}} +{{- $serviceName := include "prometheus.server.fullname" . }} +{{- $servicePort := .Values.server.service.servicePort -}} +{{- $ingressPath := .Values.server.ingress.path -}} +{{- $ingressPathType := .Values.server.ingress.pathType -}} +{{- $extraPaths := .Values.server.ingress.extraPaths -}} +apiVersion: {{ template "ingress.apiVersion" . }} +kind: Ingress +metadata: +{{- if .Values.server.ingress.annotations }} + annotations: +{{ toYaml .Values.server.ingress.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- range $key, $value := .Values.server.ingress.extraLabels }} + {{ $key }}: {{ $value }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + {{- if and $ingressSupportsIngressClassName .Values.server.ingress.ingressClassName }} + ingressClassName: {{ .Values.server.ingress.ingressClassName }} + {{- end }} + rules: + {{- range .Values.server.ingress.hosts }} + {{- $url := splitList "/" . }} + - host: {{ first $url }} + http: + paths: +{{ if $extraPaths }} +{{ toYaml $extraPaths | indent 10 }} +{{- end }} + - path: {{ $ingressPath }} + {{- if $ingressSupportsPathType }} + pathType: {{ $ingressPathType }} + {{- end }} + backend: + {{- if $ingressApiIsStable }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end -}} +{{- if .Values.server.ingress.tls }} + tls: +{{ toYaml .Values.server.ingress.tls | indent 4 }} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/netpol.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/netpol.yaml new file mode 100644 index 000000000..c8870e9ff --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/netpol.yaml @@ -0,0 +1,18 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.networkPolicy.enabled }} +apiVersion: {{ template "prometheus.networkPolicy.apiVersion" . }} +kind: NetworkPolicy +metadata: + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +spec: + podSelector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + ingress: + - ports: + - port: 9090 +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pdb.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pdb.yaml new file mode 100644 index 000000000..364cb5b49 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pdb.yaml @@ -0,0 +1,14 @@ +{{- if .Values.server.podDisruptionBudget.enabled }} +apiVersion: policy/v1beta1 +kind: PodDisruptionBudget +metadata: + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +spec: + maxUnavailable: {{ .Values.server.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + {{- include "prometheus.server.labels" . | nindent 6 }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/psp.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/psp.yaml new file mode 100644 index 000000000..e2b885f16 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/psp.yaml @@ -0,0 +1,51 @@ +{{- if and .Values.server.enabled .Values.rbac.create .Values.podSecurityPolicy.enabled }} +apiVersion: {{ template "prometheus.podSecurityPolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "prometheus.server.fullname" . }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + annotations: +{{- if .Values.server.podSecurityPolicy.annotations }} +{{ toYaml .Values.server.podSecurityPolicy.annotations | indent 4 }} +{{- end }} +spec: + privileged: false + allowPrivilegeEscalation: false + allowedCapabilities: + - 'CHOWN' + volumes: + - 'configMap' + - 'persistentVolumeClaim' + - 'emptyDir' + - 'secret' + - 'hostPath' + allowedHostPaths: + - pathPrefix: /etc + readOnly: true + - pathPrefix: {{ .Values.server.persistentVolume.mountPath }} + {{- range .Values.server.extraHostPathMounts }} + - pathPrefix: {{ .hostPath }} + readOnly: {{ .readOnly }} + {{- end }} + hostNetwork: false + hostPID: false + hostIPC: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pvc.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pvc.yaml new file mode 100644 index 000000000..a7355365c --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/pvc.yaml @@ -0,0 +1,39 @@ +{{- if .Values.server.enabled -}} +{{- if not .Values.server.statefulSet.enabled -}} +{{- if .Values.server.persistentVolume.enabled -}} +{{- if not .Values.server.persistentVolume.existingClaim -}} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + {{- if .Values.server.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.server.persistentVolume.annotations | indent 4 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + accessModes: +{{ toYaml .Values.server.persistentVolume.accessModes | indent 4 }} +{{- if .Values.server.persistentVolume.storageClass }} +{{- if (eq "-" .Values.server.persistentVolume.storageClass) }} + storageClassName: "" +{{- else }} + storageClassName: "{{ .Values.server.persistentVolume.storageClass }}" +{{- end }} +{{- end }} +{{- if .Values.server.persistentVolume.volumeBindingMode }} + volumeBindingMode: "{{ .Values.server.persistentVolume.volumeBindingMode }}" +{{- end }} + resources: + requests: + storage: "{{ .Values.server.persistentVolume.size }}" +{{- if .Values.server.persistentVolume.selector }} + selector: + {{- toYaml .Values.server.persistentVolume.selector | nindent 4 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/rolebinding.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/rolebinding.yaml new file mode 100644 index 000000000..93ce3ee13 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.server.enabled .Values.rbac.create .Values.server.useExistingClusterRoleName .Values.server.namespaces -}} +{{ range $.Values.server.namespaces -}} +--- +apiVersion: {{ template "rbac.apiVersion" $ }} +kind: RoleBinding +metadata: + labels: + {{- include "prometheus.server.labels" $ | nindent 4 }} + name: {{ template "prometheus.server.fullname" $ }} + namespace: {{ . }} +subjects: + - kind: ServiceAccount + name: {{ template "prometheus.serviceAccountName.server" $ }} +{{ include "prometheus.namespace" $ | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ $.Values.server.useExistingClusterRoleName }} +{{ end -}} +{{ end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/service.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/service.yaml new file mode 100644 index 000000000..01c5a4a8a --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/service.yaml @@ -0,0 +1,60 @@ +{{- if and .Values.server.enabled .Values.server.service.enabled -}} +apiVersion: v1 +kind: Service +metadata: +{{- if .Values.server.service.annotations }} + annotations: +{{ toYaml .Values.server.service.annotations | indent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} +{{- if .Values.server.service.labels }} +{{ toYaml .Values.server.service.labels | indent 4 }} +{{- end }} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: +{{- if .Values.server.service.clusterIP }} + clusterIP: {{ .Values.server.service.clusterIP }} +{{- end }} +{{- if .Values.server.service.externalIPs }} + externalIPs: +{{ toYaml .Values.server.service.externalIPs | indent 4 }} +{{- end }} +{{- if .Values.server.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.server.service.loadBalancerIP }} +{{- end }} +{{- if .Values.server.service.loadBalancerSourceRanges }} + loadBalancerSourceRanges: + {{- range $cidr := .Values.server.service.loadBalancerSourceRanges }} + - {{ $cidr }} + {{- end }} +{{- end }} + ports: + - name: http + port: {{ .Values.server.service.servicePort }} + protocol: TCP + targetPort: 9090 + {{- if .Values.server.service.nodePort }} + nodePort: {{ .Values.server.service.nodePort }} + {{- end }} + {{- if .Values.server.service.gRPC.enabled }} + - name: grpc + port: {{ .Values.server.service.gRPC.servicePort }} + protocol: TCP + targetPort: 10901 + {{- if .Values.server.service.gRPC.nodePort }} + nodePort: {{ .Values.server.service.gRPC.nodePort }} + {{- end }} + {{- end }} + selector: + {{- if and .Values.server.statefulSet.enabled .Values.server.service.statefulsetReplica.enabled }} + statefulset.kubernetes.io/pod-name: {{ template "prometheus.server.fullname" . }}-{{ .Values.server.service.statefulsetReplica.replica }} + {{- else -}} + {{- include "prometheus.server.matchLabels" . | nindent 4 }} +{{- if .Values.server.service.sessionAffinity }} + sessionAffinity: {{ .Values.server.service.sessionAffinity }} +{{- end }} + {{- end }} + type: "{{ .Values.server.service.type }}" +{{- end -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/serviceaccount.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/serviceaccount.yaml new file mode 100644 index 000000000..9c0502ab7 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.serviceAccounts.server.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.serviceAccountName.server" . }} +{{ include "prometheus.namespace" . | indent 2 }} + annotations: +{{ toYaml .Values.serviceAccounts.server.annotations | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/sts.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/sts.yaml new file mode 100644 index 000000000..3f76fa9ba --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/sts.yaml @@ -0,0 +1,302 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.server.statefulSet.enabled -}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: +{{- if .Values.server.statefulSet.annotations }} + annotations: + {{ toYaml .Values.server.statefulSet.annotations | nindent 4 }} +{{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + {{- if .Values.server.statefulSet.labels}} + {{ toYaml .Values.server.statefulSet.labels | nindent 4 }} + {{- end}} + name: {{ template "prometheus.server.fullname" . }} +{{ include "prometheus.namespace" . | indent 2 }} +spec: + serviceName: {{ template "prometheus.server.fullname" . }}-headless + selector: + matchLabels: + {{- include "prometheus.server.matchLabels" . | nindent 6 }} + replicas: {{ .Values.server.replicaCount }} + podManagementPolicy: {{ .Values.server.statefulSet.podManagementPolicy }} + template: + metadata: + {{- if .Values.server.podAnnotations }} + annotations: + {{ toYaml .Values.server.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "prometheus.server.labels" . | nindent 8 }} + {{- if .Values.server.podLabels}} + {{ toYaml .Values.server.podLabels | nindent 8 }} + {{- end}} + spec: +{{- if .Values.server.priorityClassName }} + priorityClassName: "{{ .Values.server.priorityClassName }}" +{{- end }} +{{- if .Values.server.schedulerName }} + schedulerName: "{{ .Values.server.schedulerName }}" +{{- end }} +{{- if semverCompare ">=1.13-0" .Capabilities.KubeVersion.GitVersion }} + {{- if or (.Values.server.enableServiceLinks) (eq (.Values.server.enableServiceLinks | toString) "") }} + enableServiceLinks: true + {{- else }} + enableServiceLinks: false + {{- end }} +{{- end }} + serviceAccountName: {{ template "prometheus.serviceAccountName.server" . }} + {{- if .Values.server.extraInitContainers }} + initContainers: +{{ toYaml .Values.server.extraInitContainers | indent 8 }} + {{- end }} + containers: + {{- if .Values.configmapReload.prometheus.enabled }} + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }}-{{ .Values.configmapReload.prometheus.name }} + image: "{{ .Values.configmapReload.prometheus.image.registry }}/{{ .Values.configmapReload.prometheus.image.repository }}:{{ .Values.configmapReload.prometheus.image.tag }}" + imagePullPolicy: "{{ .Values.configmapReload.prometheus.image.pullPolicy }}" + args: + - --volume-dir=/etc/config + - --webhook-url=http://127.0.0.1:9090{{ .Values.server.prefixURL }}/-/reload + {{- range $key, $value := .Values.configmapReload.prometheus.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraVolumeDirs }} + - --volume-dir={{ . }} + {{- end }} + {{- if .Values.configmapReload.prometheus.containerPort }} + ports: + - containerPort: {{ .Values.configmapReload.prometheus.containerPort }} + {{- end }} + resources: +{{ toYaml .Values.configmapReload.prometheus.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + readOnly: true + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- end }} + + - name: {{ template "prometheus.name" . }}-{{ .Values.server.name }} + image: "{{ .Values.server.image.registry }}/{{ .Values.server.image.repository }}:{{ .Values.server.image.tag }}" + imagePullPolicy: "{{ .Values.server.image.pullPolicy }}" + {{- if .Values.server.env }} + env: +{{ toYaml .Values.server.env | indent 12}} + {{- end }} + args: + {{- if .Values.server.defaultFlagsOverride }} + {{ toYaml .Values.server.defaultFlagsOverride | nindent 12}} + {{- else }} + {{- if .Values.server.prefixURL }} + - --web.route-prefix={{ .Values.server.prefixURL }} + {{- end }} + {{- if .Values.server.retention }} + - --storage.tsdb.retention.time={{ .Values.server.retention }} + {{- end }} + - --config.file={{ .Values.server.configPath }} + {{- if .Values.server.storagePath }} + - --storage.tsdb.path={{ .Values.server.storagePath }} + {{- else }} + - --storage.tsdb.path={{ .Values.server.persistentVolume.mountPath }} + {{- end }} + - --web.console.libraries=/etc/prometheus/console_libraries + - --web.console.templates=/etc/prometheus/consoles + {{- range .Values.server.extraFlags }} + - --{{ . }} + {{- end }} + {{- range $key, $value := .Values.server.extraArgs }} + - --{{ $key }}={{ $value }} + {{- end }} + {{- if .Values.server.baseURL }} + - --web.external-url={{ .Values.server.baseURL }} + {{- end }} + {{- end }} + ports: + - containerPort: 9090 + {{- if .Values.server.hostPort }} + hostPort: {{ .Values.server.hostPort }} + {{- end }} + readinessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/ready + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.readinessProbeInitialDelay }} + periodSeconds: {{ .Values.server.readinessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.readinessProbeTimeout }} + failureThreshold: {{ .Values.server.readinessProbeFailureThreshold }} + successThreshold: {{ .Values.server.readinessProbeSuccessThreshold }} + livenessProbe: + {{- if not .Values.server.tcpSocketProbeEnabled }} + httpGet: + path: {{ .Values.server.prefixURL }}/-/healthy + port: 9090 + scheme: {{ .Values.server.probeScheme }} + {{- else }} + tcpSocket: + port: 9090 + {{- end }} + initialDelaySeconds: {{ .Values.server.livenessProbeInitialDelay }} + periodSeconds: {{ .Values.server.livenessProbePeriodSeconds }} + timeoutSeconds: {{ .Values.server.livenessProbeTimeout }} + failureThreshold: {{ .Values.server.livenessProbeFailureThreshold }} + successThreshold: {{ .Values.server.livenessProbeSuccessThreshold }} + resources: +{{ toYaml .Values.server.resources | indent 12 }} + volumeMounts: + - name: config-volume + mountPath: /etc/config + - name: storage-volume + mountPath: {{ .Values.server.persistentVolume.mountPath }} + subPath: "{{ .Values.server.persistentVolume.subPath }}" + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + mountPath: {{ .mountPath }} + subPath: {{ .subPath }} + readOnly: {{ .readOnly }} + {{- end }} + {{- if .Values.server.extraVolumeMounts }} + {{ toYaml .Values.server.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- if .Values.server.sidecarContainers }} + {{- range $name, $spec := .Values.server.sidecarContainers }} + - name: {{ $name }} + {{- if kindIs "string" $spec }} + {{- tpl $spec $ | nindent 10 }} + {{- else }} + {{- toYaml $spec | nindent 10 }} + {{- end }} + {{- end }} + {{- end }} + hostNetwork: {{ .Values.server.hostNetwork }} + {{- if .Values.server.dnsPolicy }} + dnsPolicy: {{ .Values.server.dnsPolicy }} + {{- end }} + {{- if .Values.imagePullSecrets }} + imagePullSecrets: +{{ toYaml .Values.imagePullSecrets | indent 8 }} + {{- end }} + {{- if .Values.server.nodeSelector }} + nodeSelector: +{{ toYaml .Values.server.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.server.hostAliases }} + hostAliases: +{{ toYaml .Values.server.hostAliases | indent 8 }} + {{- end }} + {{- if .Values.server.dnsConfig }} + dnsConfig: +{{ toYaml .Values.server.dnsConfig | indent 8 }} + {{- end }} + {{- if .Values.server.securityContext }} + securityContext: +{{ toYaml .Values.server.securityContext | indent 8 }} + {{- end }} + {{- if .Values.server.tolerations }} + tolerations: +{{ toYaml .Values.server.tolerations | indent 8 }} + {{- end }} + {{- if .Values.server.affinity }} + affinity: +{{ toYaml .Values.server.affinity | indent 8 }} + {{- end }} + terminationGracePeriodSeconds: {{ .Values.server.terminationGracePeriodSeconds }} + volumes: + - name: config-volume + {{- if empty .Values.server.configFromSecret }} + configMap: + name: {{ if .Values.server.configMapOverrideName }}{{ .Release.Name }}-{{ .Values.server.configMapOverrideName }}{{- else }}{{ template "prometheus.server.fullname" . }}{{- end }} + {{- else }} + secret: + secretName: {{ .Values.server.configFromSecret }} + {{- end }} + {{- range .Values.server.extraHostPathMounts }} + - name: {{ .name }} + hostPath: + path: {{ .hostPath }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ $.Values.configmapReload.prometheus.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraConfigmapMounts }} + - name: {{ $.Values.server.name }}-{{ .name }} + configMap: + name: {{ .configMap }} + {{- end }} + {{- range .Values.server.extraSecretMounts }} + - name: {{ .name }} + secret: + secretName: {{ .secretName }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} + {{- range .Values.configmapReload.prometheus.extraConfigmapMounts }} + - name: {{ .name }} + configMap: + name: {{ .configMap }} + {{- with .optional }} + optional: {{ . }} + {{- end }} + {{- end }} +{{- if .Values.server.extraVolumes }} +{{ toYaml .Values.server.extraVolumes | indent 8}} +{{- end }} +{{- if .Values.server.persistentVolume.enabled }} + volumeClaimTemplates: + - metadata: + name: storage-volume + {{- if .Values.server.persistentVolume.annotations }} + annotations: +{{ toYaml .Values.server.persistentVolume.annotations | indent 10 }} + {{- end }} + spec: + accessModes: +{{ toYaml .Values.server.persistentVolume.accessModes | indent 10 }} + resources: + requests: + storage: "{{ .Values.server.persistentVolume.size }}" + {{- if .Values.server.persistentVolume.storageClass }} + {{- if (eq "-" .Values.server.persistentVolume.storageClass) }} + storageClassName: "" + {{- else }} + storageClassName: "{{ .Values.server.persistentVolume.storageClass }}" + {{- end }} + {{- end }} +{{- else }} + - name: storage-volume + emptyDir: + {{- if .Values.server.emptyDir.sizeLimit }} + sizeLimit: {{ .Values.server.emptyDir.sizeLimit }} + {{- else }} + {} + {{- end -}} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/vpa.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/vpa.yaml new file mode 100644 index 000000000..981a9b485 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/templates/server/vpa.yaml @@ -0,0 +1,24 @@ +{{- if .Values.server.enabled -}} +{{- if .Values.server.verticalAutoscaler.enabled -}} +apiVersion: autoscaling.k8s.io/v1beta2 +kind: VerticalPodAutoscaler +metadata: + labels: + {{- include "prometheus.server.labels" . | nindent 4 }} + name: {{ template "prometheus.server.fullname" . }}-vpa +{{ include "prometheus.namespace" . | indent 2 }} +spec: + targetRef: + apiVersion: "apps/v1" +{{- if .Values.server.statefulSet.enabled }} + kind: StatefulSet +{{- else }} + kind: Deployment +{{- end }} + name: {{ template "prometheus.server.fullname" . }} + updatePolicy: + updateMode: {{ .Values.server.verticalAutoscaler.updateMode | default "Off" | quote }} + resourcePolicy: + containerPolicies: {{ .Values.server.verticalAutoscaler.containerPolicies | default list | toYaml | trim | nindent 4 }} +{{- end -}} {{/* if .Values.server.verticalAutoscaler.enabled */}} +{{- end -}} {{/* .Values.server.enabled */}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/values.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/values.yaml new file mode 100644 index 000000000..680def88e --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/charts/prometheus/values.yaml @@ -0,0 +1,1861 @@ +rbac: + create: true + +podSecurityPolicy: + enabled: false + +imagePullSecrets: +# - name: "image-pull-secret" + +## Define serviceAccount names for components. Defaults to component's fully qualified name. +## +serviceAccounts: + alertmanager: + create: false + name: + annotations: {} + nodeExporter: + create: false + name: + annotations: {} + pushgateway: + create: false + name: + annotations: {} + server: + create: true + name: + annotations: {} + +alertmanager: + ## If false, alertmanager will not be installed + ## + enabled: true + + ## Use a ClusterRole (and ClusterRoleBinding) + ## - If set to false - we define a Role and RoleBinding in the defined namespaces ONLY + ## This makes alertmanager work - for users who do not have ClusterAdmin privs, but wants alertmanager to operate on their own namespaces, instead of clusterwide. + useClusterRole: true + + ## Set to a rolename to use existing role - skipping role creating - but still doing serviceaccount and rolebinding to the rolename set here. + useExistingRole: false + + ## alertmanager container name + ## + name: alertmanager + + ## alertmanager container image + ## + image: + registry: quay.io + repository: prometheus/alertmanager + tag: v0.23.0 + pullPolicy: IfNotPresent + + ## alertmanager priorityClassName + ## + priorityClassName: "" + + ## Custom HTTP headers for Readiness Probe + ## + ## Useful for providing HTTP Basic Auth to healthchecks + probeHeaders: [] + + ## Additional alertmanager container arguments + ## + extraArgs: {} + + ## Additional InitContainers to initialize the pod + ## + extraInitContainers: [] + + ## The URL prefix at which the container can be accessed. Useful in the case the '-web.external-url' includes a slug + ## so that the various internal URLs are still able to access as they are in the default case. + ## (Optional) + prefixURL: "" + + ## External URL which can access alertmanager + baseURL: "http://localhost:9093" + + ## Additional alertmanager container environment variable + ## For instance to add a http_proxy + ## + extraEnv: {} + + ## Additional alertmanager Secret mounts + # Defines additional mounts with secrets. Secrets must be manually created in the namespace. + extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # subPath: "" + # secretName: alertmanager-secret-files + # readOnly: true + + ## Additional alertmanager Configmap mounts + extraConfigmapMounts: [] + # - name: template-files + # mountPath: /etc/config/templates.d + # configMap: alertmanager-template-files + # readOnly: true + + ## ConfigMap override where fullname is {{.Release.Name}}-{{.Values.alertmanager.configMapOverrideName}} + ## Defining configMapOverrideName will cause templates/alertmanager-configmap.yaml + ## to NOT generate a ConfigMap resource + ## + configMapOverrideName: "" + + ## The name of a secret in the same kubernetes namespace which contains the Alertmanager config + ## Defining configFromSecret will cause templates/alertmanager-configmap.yaml + ## to NOT generate a ConfigMap resource + ## + configFromSecret: "" + + ## The configuration file name to be loaded to alertmanager + ## Must match the key within configuration loaded from ConfigMap/Secret + ## + configFileName: alertmanager.yml + + ingress: + ## If true, alertmanager Ingress will be created + ## + enabled: false + + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + + ## alertmanager Ingress annotations + ## + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: 'true' + + ## alertmanager Ingress additional labels + ## + extraLabels: {} + + ## alertmanager Ingress hostnames with optional path + ## Must be provided if Ingress is enabled + ## + hosts: [] + # - alertmanager.domain.com + # - domain.com/alertmanager + + path: / + + # pathType is only for k8s >= 1.18 + pathType: Prefix + + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + + ## alertmanager Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: [] + # - secretName: prometheus-alerts-tls + # hosts: + # - alertmanager.domain.com + + ## Alertmanager Deployment Strategy type + # strategy: + # type: Recreate + + ## Node tolerations for alertmanager scheduling to nodes with taints + ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ + ## + tolerations: [] + # - key: "key" + # operator: "Equal|Exists" + # value: "value" + # effect: "NoSchedule|PreferNoSchedule|NoExecute(1.6 only)" + + ## Node labels for alertmanager pod assignment + ## Ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Pod affinity + ## + affinity: {} + + ## PodDisruptionBudget settings + ## ref: https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ + ## + podDisruptionBudget: + enabled: false + maxUnavailable: 1 + + ## Use an alternate scheduler, e.g. "stork". + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + ## + # schedulerName: + + persistentVolume: + ## If true, alertmanager will create/use a Persistent Volume Claim + ## If false, use emptyDir + ## + enabled: true + + ## alertmanager data Persistent Volume access modes + ## Must match those of existing PV or dynamic provisioner + ## Ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ + ## + accessModes: + - ReadWriteOnce + + ## alertmanager data Persistent Volume Claim annotations + ## + annotations: {} + + ## alertmanager data Persistent Volume existing claim name + ## Requires alertmanager.persistentVolume.enabled: true + ## If defined, PVC must be created manually before volume will be bound + existingClaim: "" + + ## alertmanager data Persistent Volume mount root path + ## + mountPath: /data + + ## alertmanager data Persistent Volume size + ## + size: 2Gi + + ## alertmanager data Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + # storageClass: "-" + + ## alertmanager data Persistent Volume Binding Mode + ## If defined, volumeBindingMode: + ## If undefined (the default) or set to null, no volumeBindingMode spec is + ## set, choosing the default mode. + ## + # volumeBindingMode: "" + + ## Subdirectory of alertmanager data Persistent Volume to mount + ## Useful if the volume's root directory is not empty + ## + subPath: "" + + ## Persistent Volume Claim Selector + ## Useful if Persistent Volumes have been provisioned in advance + ## Ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#selector + ## + # selector: + # matchLabels: + # release: "stable" + # matchExpressions: + # - { key: environment, operator: In, values: [ dev ] } + + emptyDir: + ## alertmanager emptyDir volume size limit + ## + sizeLimit: "" + + ## Annotations to be added to alertmanager pods + ## + podAnnotations: {} + ## Tell prometheus to use a specific set of alertmanager pods + ## instead of all alertmanager pods found in the same namespace + ## Useful if you deploy multiple releases within the same namespace + ## + ## prometheus.io/probe: alertmanager-teamA + + ## Labels to be added to Prometheus AlertManager pods + ## + podLabels: {} + + ## Specify if a Pod Security Policy for node-exporter must be created + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + ## + podSecurityPolicy: + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + ## Use a StatefulSet if replicaCount needs to be greater than 1 (see below) + ## + replicaCount: 1 + + ## Annotations to be added to deployment + ## + deploymentAnnotations: {} + + statefulSet: + ## If true, use a statefulset instead of a deployment for pod management. + ## This allows to scale replicas to more than 1 pod + ## + enabled: false + + annotations: {} + labels: {} + podManagementPolicy: OrderedReady + + ## Alertmanager headless service to use for the statefulset + ## + headless: + annotations: {} + labels: {} + + ## Enabling peer mesh service end points for enabling the HA alert manager + ## Ref: https://github.com/prometheus/alertmanager/blob/master/README.md + enableMeshPeer: false + + servicePort: 80 + + ## alertmanager resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 100m + memory: 100Mi + requests: + cpu: 50m + memory: 50Mi + + # Custom DNS configuration to be added to alertmanager pods + dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 + + ## Security context to be added to alertmanager pods + ## + securityContext: + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + fsGroup: 65534 + + service: + annotations: {} + labels: {} + clusterIP: "" + + ## Enabling peer mesh service end points for enabling the HA alert manager + ## Ref: https://github.com/prometheus/alertmanager/blob/master/README.md + # enableMeshPeer : true + + ## List of IP addresses at which the alertmanager service is available + ## Ref: https://kubernetes.io/docs/user-guide/services/#external-ips + ## + externalIPs: [] + + loadBalancerIP: "" + loadBalancerSourceRanges: [] + servicePort: 80 + # nodePort: 30000 + sessionAffinity: None + type: ClusterIP + + ## List of initial peers + ## Ref: https://github.com/prometheus/alertmanager/blob/main/README.md#high-availability + clusterPeers: [] + +## Monitors ConfigMap changes and POSTs to a URL +## Ref: https://github.com/jimmidyson/configmap-reload +## +configmapReload: + prometheus: + ## If false, the configmap-reload container will not be deployed + ## + enabled: true + + ## configmap-reload container name + ## + name: configmap-reload + + ## configmap-reload container image + ## + image: + registry: docker.io + repository: jimmidyson/configmap-reload + tag: v0.5.0 + pullPolicy: IfNotPresent + + # containerPort: 9533 + + ## Additional configmap-reload container arguments + ## + extraArgs: {} + ## Additional configmap-reload volume directories + ## + extraVolumeDirs: [] + + + ## Additional configmap-reload mounts + ## + extraConfigmapMounts: [] + # - name: prometheus-alerts + # mountPath: /etc/alerts.d + # subPath: "" + # configMap: prometheus-alerts + # readOnly: true + + + ## configmap-reload resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 100m + memory: 100Mi + requests: + cpu: 50m + memory: 50Mi + alertmanager: + ## If false, the configmap-reload container will not be deployed + ## + enabled: true + + ## configmap-reload container name + ## + name: configmap-reload + + ## configmap-reload container image + ## + image: + registry: docker.io + repository: jimmidyson/configmap-reload + tag: v0.5.0 + pullPolicy: IfNotPresent + + # containerPort: 9533 + + ## Additional configmap-reload container arguments + ## + extraArgs: {} + ## Additional configmap-reload volume directories + ## + extraVolumeDirs: [] + + + ## Additional configmap-reload mounts + ## + extraConfigmapMounts: [] + # - name: prometheus-alerts + # mountPath: /etc/alerts.d + # subPath: "" + # configMap: prometheus-alerts + # readOnly: true + + + ## configmap-reload resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 100m + memory: 100Mi + requests: + cpu: 50m + memory: 50Mi + +kubeStateMetrics: + ## If false, kube-state-metrics sub-chart will not be installed + ## + enabled: true + +## kube-state-metrics sub-chart configurable values +## Please see https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics +## +# kube-state-metrics: + +nodeExporter: + ## If false, node-exporter will not be installed + ## + enabled: true + + ## If true, node-exporter pods share the host network namespace + ## + hostNetwork: true + + ## If true, node-exporter pods share the host PID namespace + ## + hostPID: true + + ## If true, node-exporter pods mounts host / at /host/root + ## + hostRootfs: true + + ## node-exporter container name + ## + name: node-exporter + + ## node-exporter container image + ## + image: + registry: quay.io + repository: prometheus/node-exporter + tag: v1.3.0 + pullPolicy: IfNotPresent + + ## Specify if a Pod Security Policy for node-exporter must be created + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + ## + podSecurityPolicy: + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + ## node-exporter priorityClassName + ## + priorityClassName: "" + + ## Custom Update Strategy + ## + updateStrategy: + type: RollingUpdate + + ## Additional node-exporter container arguments + ## + extraArgs: {} + + ## Additional InitContainers to initialize the pod + ## + extraInitContainers: [] + + ## Additional node-exporter hostPath mounts + ## + extraHostPathMounts: [] + # - name: textfile-dir + # mountPath: /srv/txt_collector + # hostPath: /var/lib/node-exporter + # readOnly: true + # mountPropagation: HostToContainer + + extraConfigmapMounts: [] + # - name: certs-configmap + # mountPath: /prometheus + # configMap: certs-configmap + # readOnly: true + + ## Node tolerations for node-exporter scheduling to nodes with taints + ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ + ## + tolerations: [] + # - key: "key" + # operator: "Equal|Exists" + # value: "value" + # effect: "NoSchedule|PreferNoSchedule|NoExecute(1.6 only)" + + ## Node labels for node-exporter pod assignment + ## Ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Annotations to be added to node-exporter pods + ## + podAnnotations: {} + + ## Labels to be added to node-exporter pods + ## + pod: + labels: {} + + ## PodDisruptionBudget settings + ## ref: https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ + ## + podDisruptionBudget: + enabled: false + maxUnavailable: 1 + + ## node-exporter resource limits & requests + ## Ref: https://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 500m + memory: 100Mi + requests: + cpu: 200m + memory: 50Mi + container: + securityContext: + allowPrivilegeEscalation: false + # Custom DNS configuration to be added to node-exporter pods + dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 + + ## Security context to be added to node-exporter pods + ## + securityContext: + fsGroup: 65534 + runAsGroup: 65534 + runAsNonRoot: true + runAsUser: 65534 + + service: + annotations: + prometheus.io/scrape: "true" + labels: {} + + # Exposed as a headless service: + # https://kubernetes.io/docs/concepts/services-networking/service/#headless-services + clusterIP: "" + + ## List of IP addresses at which the node-exporter service is available + ## Ref: https://kubernetes.io/docs/user-guide/services/#external-ips + ## + externalIPs: [] + + hostPort: 9100 + loadBalancerIP: "" + loadBalancerSourceRanges: [] + servicePort: 9100 + type: ClusterIP + +server: + ## Prometheus server container name + ## + enabled: true + + ## Use a ClusterRole (and ClusterRoleBinding) + ## - If set to false - we define a RoleBinding in the defined namespaces ONLY + ## + ## NB: because we need a Role with nonResourceURL's ("/metrics") - you must get someone with Cluster-admin privileges to define this role for you, before running with this setting enabled. + ## This makes prometheus work - for users who do not have ClusterAdmin privs, but wants prometheus to operate on their own namespaces, instead of clusterwide. + ## + ## You MUST also set namespaces to the ones you have access to and want monitored by Prometheus. + ## + # useExistingClusterRoleName: nameofclusterrole + + ## namespaces to monitor (instead of monitoring all - clusterwide). Needed if you want to run without Cluster-admin privileges. + # namespaces: + # - yournamespace + + name: server + + # sidecarContainers - add more containers to prometheus server + # Key/Value where Key is the sidecar `- name: ` + # Example: + # sidecarContainers: + # webserver: + # image: nginx + sidecarContainers: {} + + # sidecarTemplateValues - context to be used in template for sidecarContainers + # Example: + # sidecarTemplateValues: *your-custom-globals + # sidecarContainers: + # webserver: |- + # {{ include "webserver-container-template" . }} + # Template for `webserver-container-template` might looks like this: + # image: "{{ .Values.server.sidecarTemplateValues.repository }}:{{ .Values.server.sidecarTemplateValues.tag }}" + # ... + # + sidecarTemplateValues: {} + + ## Prometheus server container image + ## + image: + registry: quay.io + repository: prometheus/prometheus + tag: v2.34.0 + pullPolicy: IfNotPresent + + ## prometheus server priorityClassName + ## + priorityClassName: "" + + ## EnableServiceLinks indicates whether information about services should be injected + ## into pod's environment variables, matching the syntax of Docker links. + ## WARNING: the field is unsupported and will be skipped in K8s prior to v1.13.0. + ## + enableServiceLinks: true + + ## The URL prefix at which the container can be accessed. Useful in the case the '-web.external-url' includes a slug + ## so that the various internal URLs are still able to access as they are in the default case. + ## (Optional) + prefixURL: "" + + ## External URL which can access prometheus + ## Maybe same with Ingress host name + baseURL: "" + + ## Additional server container environment variables + ## + ## You specify this manually like you would a raw deployment manifest. + ## This means you can bind in environment variables from secrets. + ## + ## e.g. static environment variable: + ## - name: DEMO_GREETING + ## value: "Hello from the environment" + ## + ## e.g. secret environment variable: + ## - name: USERNAME + ## valueFrom: + ## secretKeyRef: + ## name: mysecret + ## key: username + env: [] + + # List of flags to override default parameters, e.g: + # - --enable-feature=agent + # - --storage.agent.retention.max-time=30m + defaultFlagsOverride: [] + + extraFlags: + - web.enable-lifecycle + ## web.enable-admin-api flag controls access to the administrative HTTP API which includes functionality such as + ## deleting time series. This is disabled by default. + # - web.enable-admin-api + ## + ## storage.tsdb.no-lockfile flag controls BD locking + # - storage.tsdb.no-lockfile + ## + ## storage.tsdb.wal-compression flag enables compression of the write-ahead log (WAL) + # - storage.tsdb.wal-compression + + ## Path to a configuration file on prometheus server container FS + configPath: /etc/config/prometheus.yml + + ### The data directory used by prometheus to set --storage.tsdb.path + ### When empty server.persistentVolume.mountPath is used instead + storagePath: "" + + global: + ## How frequently to scrape targets by default + ## + scrape_interval: 1m + ## How long until a scrape request times out + ## + scrape_timeout: 10s + ## How frequently to evaluate rules + ## + evaluation_interval: 1m + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write + ## + remoteWrite: [] + ## https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_read + ## + remoteRead: [] + + ## Custom HTTP headers for Liveness/Readiness/Startup Probe + ## + ## Useful for providing HTTP Basic Auth to healthchecks + probeHeaders: [] + + ## Additional Prometheus server container arguments + ## + extraArgs: {} + + ## Additional InitContainers to initialize the pod + ## + extraInitContainers: [] + + ## Additional Prometheus server Volume mounts + ## + extraVolumeMounts: [] + + ## Additional Prometheus server Volumes + ## + extraVolumes: [] + + ## Additional Prometheus server hostPath mounts + ## + extraHostPathMounts: [] + # - name: certs-dir + # mountPath: /etc/kubernetes/certs + # subPath: "" + # hostPath: /etc/kubernetes/certs + # readOnly: true + + extraConfigmapMounts: [] + # - name: certs-configmap + # mountPath: /prometheus + # subPath: "" + # configMap: certs-configmap + # readOnly: true + + ## Additional Prometheus server Secret mounts + # Defines additional mounts with secrets. Secrets must be manually created in the namespace. + extraSecretMounts: [] + # - name: secret-files + # mountPath: /etc/secrets + # subPath: "" + # secretName: prom-secret-files + # readOnly: true + + ## ConfigMap override where fullname is {{.Release.Name}}-{{.Values.server.configMapOverrideName}} + ## Defining configMapOverrideName will cause templates/server-configmap.yaml + ## to NOT generate a ConfigMap resource + ## + configMapOverrideName: "" + + ingress: + ## If true, Prometheus server Ingress will be created + ## + enabled: false + + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + + ## Prometheus server Ingress annotations + ## + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: 'true' + + ## Prometheus server Ingress additional labels + ## + extraLabels: {} + + ## Prometheus server Ingress hostnames with optional path + ## Must be provided if Ingress is enabled + ## + hosts: [] + # - prometheus.domain.com + # - domain.com/prometheus + + path: / + + # pathType is only for k8s >= 1.18 + pathType: Prefix + + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + + ## Prometheus server Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: [] + # - secretName: prometheus-server-tls + # hosts: + # - prometheus.domain.com + + ## Server Deployment Strategy type + strategy: + type: RollingUpdate + + ## hostAliases allows adding entries to /etc/hosts inside the containers + hostAliases: [] + # - ip: "127.0.0.1" + # hostnames: + # - "example.com" + + ## Node tolerations for server scheduling to nodes with taints + ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ + ## + tolerations: [] + # - key: "key" + # operator: "Equal|Exists" + # value: "value" + # effect: "NoSchedule|PreferNoSchedule|NoExecute(1.6 only)" + + ## Node labels for Prometheus server pod assignment + ## Ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Pod affinity + ## + affinity: {} + + ## PodDisruptionBudget settings + ## ref: https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ + ## + podDisruptionBudget: + enabled: false + maxUnavailable: 1 + + ## Use an alternate scheduler, e.g. "stork". + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + ## + # schedulerName: + + persistentVolume: + ## If true, Prometheus server will create/use a Persistent Volume Claim + ## If false, use emptyDir + ## + enabled: true + + ## Prometheus server data Persistent Volume access modes + ## Must match those of existing PV or dynamic provisioner + ## Ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ + ## + accessModes: + - ReadWriteOnce + + ## Prometheus server data Persistent Volume annotations + ## + annotations: {} + + ## Prometheus server data Persistent Volume existing claim name + ## Requires server.persistentVolume.enabled: true + ## If defined, PVC must be created manually before volume will be bound + existingClaim: "" + + ## Prometheus server data Persistent Volume mount root path + ## + mountPath: /data + + ## Prometheus server data Persistent Volume size + ## + size: 8Gi + + ## Prometheus server data Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + # storageClass: "-" + + ## Prometheus server data Persistent Volume Binding Mode + ## If defined, volumeBindingMode: + ## If undefined (the default) or set to null, no volumeBindingMode spec is + ## set, choosing the default mode. + ## + # volumeBindingMode: "" + + ## Subdirectory of Prometheus server data Persistent Volume to mount + ## Useful if the volume's root directory is not empty + ## + subPath: "" + + ## Persistent Volume Claim Selector + ## Useful if Persistent Volumes have been provisioned in advance + ## Ref: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#selector + ## + # selector: + # matchLabels: + # release: "stable" + # matchExpressions: + # - { key: environment, operator: In, values: [ dev ] } + + emptyDir: + ## Prometheus server emptyDir volume size limit + ## + sizeLimit: "" + + ## Annotations to be added to Prometheus server pods + ## + podAnnotations: {} + # iam.amazonaws.com/role: prometheus + + ## Labels to be added to Prometheus server pods + ## + podLabels: {} + + ## Prometheus AlertManager configuration + ## + alertmanagers: [] + + ## Specify if a Pod Security Policy for node-exporter must be created + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + ## + podSecurityPolicy: + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + ## Use a StatefulSet if replicaCount needs to be greater than 1 (see below) + ## + replicaCount: 1 + + ## Annotations to be added to deployment + ## + deploymentAnnotations: {} + + statefulSet: + ## If true, use a statefulset instead of a deployment for pod management. + ## This allows to scale replicas to more than 1 pod + ## + enabled: false + + annotations: {} + labels: {} + podManagementPolicy: OrderedReady + + ## Alertmanager headless service to use for the statefulset + ## + headless: + annotations: {} + labels: {} + servicePort: 80 + ## Enable gRPC port on service to allow auto discovery with thanos-querier + gRPC: + enabled: false + servicePort: 10901 + # nodePort: 10901 + + ## Prometheus server readiness and liveness probe initial delay and timeout + ## Ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/ + ## + tcpSocketProbeEnabled: false + probeScheme: HTTP + readinessProbeInitialDelay: 30 + readinessProbePeriodSeconds: 5 + readinessProbeTimeout: 4 + readinessProbeFailureThreshold: 3 + readinessProbeSuccessThreshold: 1 + livenessProbeInitialDelay: 30 + livenessProbePeriodSeconds: 15 + livenessProbeTimeout: 10 + livenessProbeFailureThreshold: 3 + livenessProbeSuccessThreshold: 1 + startupProbe: + enabled: false + periodSeconds: 5 + failureThreshold: 30 + timeoutSeconds: 10 + + ## Prometheus server resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 1000m + memory: 512Mi + requests: + cpu: 500m + memory: 512Mi + + # Required for use in managed kubernetes clusters (such as AWS EKS) with custom CNI (such as calico), + # because control-plane managed by AWS cannot communicate with pods' IP CIDR and admission webhooks are not working + ## + hostNetwork: false + + # When hostNetwork is enabled, you probably want to set this to ClusterFirstWithHostNet + dnsPolicy: ClusterFirst + + # Use hostPort + # hostPort: 9090 + + ## Vertical Pod Autoscaler config + ## Ref: https://github.com/kubernetes/autoscaler/tree/master/vertical-pod-autoscaler + verticalAutoscaler: + ## If true a VPA object will be created for the controller (either StatefulSet or Deployemnt, based on above configs) + enabled: false + # updateMode: "Auto" + # containerPolicies: + # - containerName: 'prometheus-server' + + # Custom DNS configuration to be added to prometheus server pods + dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 + ## Security context to be added to server pods + ## + securityContext: + runAsUser: 65534 + runAsNonRoot: true + runAsGroup: 65534 + fsGroup: 65534 + + service: + ## If false, no Service will be created for the Prometheus server + ## + enabled: true + + annotations: {} + labels: {} + clusterIP: "" + + ## List of IP addresses at which the Prometheus server service is available + ## Ref: https://kubernetes.io/docs/user-guide/services/#external-ips + ## + externalIPs: [] + + loadBalancerIP: "" + loadBalancerSourceRanges: [] + servicePort: 80 + sessionAffinity: None + type: ClusterIP + + ## Enable gRPC port on service to allow auto discovery with thanos-querier + gRPC: + enabled: false + servicePort: 10901 + # nodePort: 10901 + + ## If using a statefulSet (statefulSet.enabled=true), configure the + ## service to connect to a specific replica to have a consistent view + ## of the data. + statefulsetReplica: + enabled: false + replica: 0 + + ## Prometheus server pod termination grace period + ## + terminationGracePeriodSeconds: 300 + + ## Prometheus data retention period (default if not specified is 15 days) + ## + retention: "15d" + + ## Array of extra Kubernetes manifests, if you want to deploy + extraObjects: [] + +pushgateway: + ## If false, pushgateway will not be installed + ## + enabled: true + + ## Use an alternate scheduler, e.g. "stork". + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + ## + # schedulerName: + + ## pushgateway container name + ## + name: pushgateway + + ## pushgateway container image + ## + image: + registry: docker.io + repository: prom/pushgateway + tag: v1.4.2 + pullPolicy: IfNotPresent + + ## pushgateway priorityClassName + ## + priorityClassName: "" + + ## Additional pushgateway container arguments + ## + ## for example: persistence.file: /data/pushgateway.data + extraArgs: {} + + ## Additional InitContainers to initialize the pod + ## + extraInitContainers: [] + + ingress: + ## If true, pushgateway Ingress will be created + ## + enabled: false + + # For Kubernetes >= 1.18 you should specify the ingress-controller via the field ingressClassName + # See https://kubernetes.io/blog/2020/04/02/improvements-to-the-ingress-api-in-kubernetes-1.18/#specifying-the-class-of-an-ingress + # ingressClassName: nginx + + ## pushgateway Ingress annotations + ## + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: 'true' + + ## pushgateway Ingress hostnames with optional path + ## Must be provided if Ingress is enabled + ## + hosts: [] + # - pushgateway.domain.com + # - domain.com/pushgateway + + path: / + + # pathType is only for k8s >= 1.18 + pathType: Prefix + + ## Extra paths to prepend to every host configuration. This is useful when working with annotation based services. + extraPaths: [] + # - path: /* + # backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + + ## pushgateway Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: [] + # - secretName: prometheus-alerts-tls + # hosts: + # - pushgateway.domain.com + + ## Node tolerations for pushgateway scheduling to nodes with taints + ## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ + ## + tolerations: [] + # - key: "key" + # operator: "Equal|Exists" + # value: "value" + # effect: "NoSchedule|PreferNoSchedule|NoExecute(1.6 only)" + + ## Node labels for pushgateway pod assignment + ## Ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Annotations to be added to pushgateway pods + ## + podAnnotations: {} + + ## Labels to be added to pushgateway pods + ## + podLabels: {} + + ## Specify if a Pod Security Policy for node-exporter must be created + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ + ## + podSecurityPolicy: + annotations: {} + ## Specify pod annotations + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#apparmor + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#seccomp + ## Ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/#sysctl + ## + # seccomp.security.alpha.kubernetes.io/allowedProfileNames: '*' + # seccomp.security.alpha.kubernetes.io/defaultProfileName: 'docker/default' + # apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default' + + replicaCount: 1 + + ## Annotations to be added to deployment + ## + deploymentAnnotations: {} + + ## PodDisruptionBudget settings + ## ref: https://kubernetes.io/docs/concepts/workloads/pods/disruptions/ + ## + podDisruptionBudget: + enabled: false + maxUnavailable: 1 + + ## pushgateway resource requests and limits + ## Ref: http://kubernetes.io/docs/user-guide/compute-resources/ + ## + resources: + limits: + cpu: 100m + memory: 100Mi + requests: + cpu: 50m + memory: 50Mi + + ## Vertical Pod Autoscaler config + ## Ref: https://github.com/kubernetes/autoscaler/tree/master/vertical-pod-autoscaler + verticalAutoscaler: + ## If true a VPA object will be created for the controller + enabled: false + # updateMode: "Auto" + # containerPolicies: + # - containerName: 'prometheus-pushgateway' + + # Custom DNS configuration to be added to push-gateway pods + dnsConfig: {} + # nameservers: + # - 1.2.3.4 + # searches: + # - ns1.svc.cluster-domain.example + # - my.dns.search.suffix + # options: + # - name: ndots + # value: "2" + # - name: edns0 + + ## Security context to be added to push-gateway pods + ## + securityContext: + runAsUser: 65534 + runAsNonRoot: true + + service: + annotations: + prometheus.io/probe: pushgateway + labels: {} + clusterIP: "" + + ## List of IP addresses at which the pushgateway service is available + ## Ref: https://kubernetes.io/docs/user-guide/services/#external-ips + ## + externalIPs: [] + + loadBalancerIP: "" + loadBalancerSourceRanges: [] + servicePort: 9091 + type: ClusterIP + + ## pushgateway Deployment Strategy type + # strategy: + # type: Recreate + + persistentVolume: + ## If true, pushgateway will create/use a Persistent Volume Claim + ## + enabled: false + + ## pushgateway data Persistent Volume access modes + ## Must match those of existing PV or dynamic provisioner + ## Ref: http://kubernetes.io/docs/user-guide/persistent-volumes/ + ## + accessModes: + - ReadWriteOnce + + ## pushgateway data Persistent Volume Claim annotations + ## + annotations: {} + + ## pushgateway data Persistent Volume existing claim name + ## Requires pushgateway.persistentVolume.enabled: true + ## If defined, PVC must be created manually before volume will be bound + existingClaim: "" + + ## pushgateway data Persistent Volume mount root path + ## + mountPath: /data + + ## pushgateway data Persistent Volume size + ## + size: 2Gi + + ## pushgateway data Persistent Volume Storage Class + ## If defined, storageClassName: + ## If set to "-", storageClassName: "", which disables dynamic provisioning + ## If undefined (the default) or set to null, no storageClassName spec is + ## set, choosing the default provisioner. (gp2 on AWS, standard on + ## GKE, AWS & OpenStack) + ## + # storageClass: "-" + + ## pushgateway data Persistent Volume Binding Mode + ## If defined, volumeBindingMode: + ## If undefined (the default) or set to null, no volumeBindingMode spec is + ## set, choosing the default mode. + ## + # volumeBindingMode: "" + + ## Subdirectory of pushgateway data Persistent Volume to mount + ## Useful if the volume's root directory is not empty + ## + subPath: "" + + +## alertmanager ConfigMap entries +## +alertmanagerFiles: + alertmanager.yml: + global: {} + # slack_api_url: '' + + receivers: + - name: default-receiver + # slack_configs: + # - channel: '@you' + # send_resolved: true + + route: + group_wait: 10s + group_interval: 5m + receiver: default-receiver + repeat_interval: 3h + +## Prometheus server ConfigMap entries for rule files (allow prometheus labels interpolation) +ruleFiles: {} + +## Prometheus server ConfigMap entries +## +serverFiles: + + ## Alerts configuration + ## Ref: https://prometheus.io/docs/prometheus/latest/configuration/alerting_rules/ + alerting_rules.yml: {} + # groups: + # - name: Instances + # rules: + # - alert: InstanceDown + # expr: up == 0 + # for: 5m + # labels: + # severity: page + # annotations: + # description: '{{ $labels.instance }} of job {{ $labels.job }} has been down for more than 5 minutes.' + # summary: 'Instance {{ $labels.instance }} down' + ## DEPRECATED DEFAULT VALUE, unless explicitly naming your files, please use alerting_rules.yml + alerts: {} + + ## Records configuration + ## Ref: https://prometheus.io/docs/prometheus/latest/configuration/recording_rules/ + recording_rules.yml: {} + ## DEPRECATED DEFAULT VALUE, unless explicitly naming your files, please use recording_rules.yml + rules: {} + + prometheus.yml: + rule_files: + - /etc/config/recording_rules.yml + - /etc/config/alerting_rules.yml + ## Below two files are DEPRECATED will be removed from this default values file + - /etc/config/rules + - /etc/config/alerts + + scrape_configs: + - job_name: prometheus + static_configs: + - targets: + - localhost:9090 + + # A scrape configuration for running Prometheus on a Kubernetes cluster. + # This uses separate scrape configs for cluster components (i.e. API server, node) + # and services to allow each to use different authentication configs. + # + # Kubernetes labels will be added as Prometheus labels on metrics via the + # `labelmap` relabeling action. + + # Scrape config for API servers. + # + # Kubernetes exposes API servers as endpoints to the default/kubernetes + # service so this uses `endpoints` role and uses relabelling to only keep + # the endpoints associated with the default/kubernetes service using the + # default named port `https`. This works for single API server deployments as + # well as HA API server deployments. + - job_name: 'kubernetes-apiservers' + + kubernetes_sd_configs: + - role: endpoints + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + # Keep only the default/kubernetes service endpoints for the https port. This + # will add targets for each API server which Kubernetes adds an endpoint to + # the default/kubernetes service. + relabel_configs: + - source_labels: [__meta_kubernetes_namespace, __meta_kubernetes_service_name, __meta_kubernetes_endpoint_port_name] + action: keep + regex: default;kubernetes;https + + - job_name: 'kubernetes-nodes' + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + kubernetes_sd_configs: + - role: node + + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_node_label_(.+) + - target_label: __address__ + replacement: kubernetes.default.svc:443 + - source_labels: [__meta_kubernetes_node_name] + regex: (.+) + target_label: __metrics_path__ + replacement: /api/v1/nodes/$1/proxy/metrics + + + - job_name: 'kubernetes-nodes-cadvisor' + + # Default to scraping over https. If required, just disable this or change to + # `http`. + scheme: https + + # This TLS & bearer token file config is used to connect to the actual scrape + # endpoints for cluster components. This is separate to discovery auth + # configuration because discovery & scraping are two separate concerns in + # Prometheus. The discovery auth config is automatic if Prometheus runs inside + # the cluster. Otherwise, more config options have to be provided within the + # . + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + # If your node certificates are self-signed or use a different CA to the + # master CA, then disable certificate verification below. Note that + # certificate verification is an integral part of a secure infrastructure + # so this should only be disabled in a controlled environment. You can + # disable certificate verification by uncommenting the line below. + # + insecure_skip_verify: true + bearer_token_file: /var/run/secrets/kubernetes.io/serviceaccount/token + + kubernetes_sd_configs: + - role: node + + # This configuration will work only on kubelet 1.7.3+ + # As the scrape endpoints for cAdvisor have changed + # if you are using older version you need to change the replacement to + # replacement: /api/v1/nodes/$1:4194/proxy/metrics + # more info here https://github.com/coreos/prometheus-operator/issues/633 + relabel_configs: + - action: labelmap + regex: __meta_kubernetes_node_label_(.+) + - target_label: __address__ + replacement: kubernetes.default.svc:443 + - source_labels: [__meta_kubernetes_node_name] + regex: (.+) + target_label: __metrics_path__ + replacement: /api/v1/nodes/$1/proxy/metrics/cadvisor + + # Scrape config for service endpoints. + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/scrape`: Only scrape services that have a value of + # `true`, except if `prometheus.io/scrape-slow` is set to `true` as well. + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: If the metrics are exposed on a different port to the + # service then set this appropriately. + # * `prometheus.io/param_`: If the metrics endpoint uses parameters + # then you can set any parameter + - job_name: 'kubernetes-service-endpoints' + honor_labels: true + + kubernetes_sd_configs: + - role: endpoints + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape] + action: keep + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape_slow] + action: drop + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scheme] + action: replace + target_label: __scheme__ + regex: (https?) + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_service_annotation_prometheus_io_port] + action: replace + target_label: __address__ + regex: ([^:]+)(?::\d+)?;(\d+) + replacement: $1:$2 + - action: labelmap + regex: __meta_kubernetes_service_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + action: replace + target_label: service + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + + # Scrape config for slow service endpoints; same as above, but with a larger + # timeout and a larger interval + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/scrape-slow`: Only scrape services that have a value of `true` + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: If the metrics are exposed on a different port to the + # service then set this appropriately. + # * `prometheus.io/param_`: If the metrics endpoint uses parameters + # then you can set any parameter + - job_name: 'kubernetes-service-endpoints-slow' + honor_labels: true + + scrape_interval: 5m + scrape_timeout: 30s + + kubernetes_sd_configs: + - role: endpoints + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scrape_slow] + action: keep + regex: true + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_scheme] + action: replace + target_label: __scheme__ + regex: (https?) + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_service_annotation_prometheus_io_port] + action: replace + target_label: __address__ + regex: ([^:]+)(?::\d+)?;(\d+) + replacement: $1:$2 + - action: labelmap + regex: __meta_kubernetes_service_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + action: replace + target_label: service + - source_labels: [__meta_kubernetes_pod_node_name] + action: replace + target_label: node + + - job_name: 'prometheus-pushgateway' + honor_labels: true + + kubernetes_sd_configs: + - role: service + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_probe] + action: keep + regex: pushgateway + + # Example scrape config for probing services via the Blackbox Exporter. + # + # The relabeling allows the actual service scrape endpoint to be configured + # via the following annotations: + # + # * `prometheus.io/probe`: Only probe services that have a value of `true` + - job_name: 'kubernetes-services' + honor_labels: true + + metrics_path: /probe + params: + module: [http_2xx] + + kubernetes_sd_configs: + - role: service + + relabel_configs: + - source_labels: [__meta_kubernetes_service_annotation_prometheus_io_probe] + action: keep + regex: true + - source_labels: [__address__] + target_label: __param_target + - target_label: __address__ + replacement: blackbox + - source_labels: [__param_target] + target_label: instance + - action: labelmap + regex: __meta_kubernetes_service_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + target_label: namespace + - source_labels: [__meta_kubernetes_service_name] + target_label: service + + # Example scrape config for pods + # + # The relabeling allows the actual pod scrape endpoint to be configured via the + # following annotations: + # + # * `prometheus.io/scrape`: Only scrape pods that have a value of `true`, + # except if `prometheus.io/scrape-slow` is set to `true` as well. + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: Scrape the pod on the indicated port instead of the default of `9102`. + - job_name: 'kubernetes-pods' + honor_labels: true + + kubernetes_sd_configs: + - role: pod + + relabel_configs: + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape] + action: keep + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape_slow] + action: drop + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scheme] + action: replace + regex: (https?) + target_label: __scheme__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_pod_annotation_prometheus_io_port] + action: replace + regex: ([^:]+)(?::\d+)?;(\d+) + replacement: $1:$2 + target_label: __address__ + - action: labelmap + regex: __meta_kubernetes_pod_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_pod_name] + action: replace + target_label: pod + - source_labels: [__meta_kubernetes_pod_phase] + regex: Pending|Succeeded|Failed|Completed + action: drop + + # Example Scrape config for pods which should be scraped slower. An useful example + # would be stackriver-exporter which queries an API on every scrape of the pod + # + # The relabeling allows the actual pod scrape endpoint to be configured via the + # following annotations: + # + # * `prometheus.io/scrape-slow`: Only scrape pods that have a value of `true` + # * `prometheus.io/scheme`: If the metrics endpoint is secured then you will need + # to set this to `https` & most likely set the `tls_config` of the scrape config. + # * `prometheus.io/path`: If the metrics path is not `/metrics` override this. + # * `prometheus.io/port`: Scrape the pod on the indicated port instead of the default of `9102`. + - job_name: 'kubernetes-pods-slow' + honor_labels: true + + scrape_interval: 5m + scrape_timeout: 30s + + kubernetes_sd_configs: + - role: pod + + relabel_configs: + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scrape_slow] + action: keep + regex: true + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_scheme] + action: replace + regex: (https?) + target_label: __scheme__ + - source_labels: [__meta_kubernetes_pod_annotation_prometheus_io_path] + action: replace + target_label: __metrics_path__ + regex: (.+) + - source_labels: [__address__, __meta_kubernetes_pod_annotation_prometheus_io_port] + action: replace + regex: ([^:]+)(?::\d+)?;(\d+) + replacement: $1:$2 + target_label: __address__ + - action: labelmap + regex: __meta_kubernetes_pod_annotation_prometheus_io_param_(.+) + replacement: __param_$1 + - action: labelmap + regex: __meta_kubernetes_pod_label_(.+) + - source_labels: [__meta_kubernetes_namespace] + action: replace + target_label: namespace + - source_labels: [__meta_kubernetes_pod_name] + action: replace + target_label: pod + - source_labels: [__meta_kubernetes_pod_phase] + regex: Pending|Succeeded|Failed|Completed + action: drop + +# adds additional scrape configs to prometheus.yml +# must be a string so you have to add a | after extraScrapeConfigs: +# example adds prometheus-blackbox-exporter scrape config +extraScrapeConfigs: + # - job_name: 'prometheus-blackbox-exporter' + # metrics_path: /probe + # params: + # module: [http_2xx] + # static_configs: + # - targets: + # - https://example.com + # relabel_configs: + # - source_labels: [__address__] + # target_label: __param_target + # - source_labels: [__param_target] + # target_label: instance + # - target_label: __address__ + # replacement: prometheus-blackbox-exporter:9115 + +# Adds option to add alert_relabel_configs to avoid duplicate alerts in alertmanager +# useful in H/A prometheus with different external labels but the same alerts +alertRelabelConfigs: + # alert_relabel_configs: + # - source_labels: [dc] + # regex: (.+)\d+ + # target_label: dc + +networkPolicy: + ## Enable creation of NetworkPolicy resources. + ## + enabled: false + +# Force namespace of namespaced resources +forceNamespace: null diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/_helpers.tpl b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/_helpers.tpl new file mode 100644 index 000000000..4dae8715b --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/_helpers.tpl @@ -0,0 +1,49 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "monitoring.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 "monitoring.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 "monitoring.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Override the naming defined by the prometheus chart. +Added as a fix for https://github.com/grafana/loki/issues/1169 +*/}} +{{- define "prometheus.fullname" -}} +{{- if .Values.prometheus.server.fullnameOverride -}} +{{- .Values.prometheus.server.fullnameOverride | trunc 63 -}} +{{- 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 -}} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/datasources.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/datasources.yaml new file mode 100644 index 000000000..3300d7aec --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/monitoring/templates/datasources.yaml @@ -0,0 +1,25 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "monitoring.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + app: {{ template "monitoring.name" . }} + chart: {{ template "monitoring.chart" . }} + release: {{ .Release.Name }} + heritage: {{ .Release.Service }} + app.kubernetes.io/instance: {{ template "monitoring.name" . }} + {{- include "k8s-triliovault-operator.observability" . | nindent 4 }} + grafana_datasource: "1" +data: + monitoring-datasource.yaml: |- + apiVersion: 1 + datasources: +{{- if .Values.prometheus.enabled }} + - name: Prometheus + type: prometheus + access: proxy + isDefault: true + url: http://{{ include "prometheus.fullname" .}}:{{ .Values.prometheus.server.service.servicePort }}{{ .Values.prometheus.server.prefixURL }} + version: 1 +{{- end }} diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/Chart.yaml new file mode 100644 index 000000000..12fa0fc95 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/Chart.yaml @@ -0,0 +1,16 @@ +apiVersion: v2 +appVersion: 0.1.0 +dependencies: +- condition: grafana.enabled + name: grafana + repository: https://grafana.github.io/helm-charts + version: ^6.29.2 +description: Visualization Stack designed to manage the K8s-TrilioVault Application's + Visualization. +icon: https://www.trilio.io/wp-content/uploads/2021/01/Trilio-2020-logo-RGB-gray-green.png +kubeVersion: '>=1.19.0-0' +maintainers: +- email: support@trilio.io + name: Trilio +name: visualization +version: 0.1.0 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/Chart.yaml b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/Chart.yaml new file mode 100644 index 000000000..5a3ffe454 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/Chart.yaml @@ -0,0 +1,14 @@ +apiVersion: v2 +appVersion: 8.5.0 +description: The leading tool for querying and visualizing time series and metrics. +home: https://grafana.net +icon: https://raw.githubusercontent.com/grafana/grafana/master/public/img/logo_transparent_400x.png +kubeVersion: ^1.8.0-0 +maintainers: +- email: support@trilio.io + name: Trilio +name: grafana +sources: +- https://github.com/grafana/grafana +type: application +version: 6.29.2 diff --git a/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-detail.json b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-detail.json new file mode 100644 index 000000000..cee756e93 --- /dev/null +++ b/charts/trilio/k8s-triliovault-operator/5.0.0/charts/observability/charts/visualization/charts/grafana/dashboards/backup-detail.json @@ -0,0 +1,926 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + }, + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + } + ], + "__elements": [], + "__requires": [ + { + "type": "panel", + "id": "gauge", + "name": "Gauge", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "8.5.0" + }, + { + "type": "panel", + "id": "logs", + "name": "Logs", + "version": "" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "table-old", + "name": "Table (old)", + "version": "" + }, + { + "type": "panel", + "id": "text", + "name": "Text", + "version": "" + } + ], + "annotations": { + "list": [ + { + "$$hashKey": "object:20", + "builtIn": 1, + "datasource": { + "type": "datasource", + "uid": "grafana" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "target": { + "limit": 100, + "matchAny": false, + "tags": [], + "type": "dashboard" + }, + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "gnetId": 12601, + "graphTooltip": 0, + "id": null, + "iteration": 1655448146244, + "links": [], + "liveNow": false, + "panels": [ + { + "datasource": { + "uid": "${DS_PROMETHEUS}" + }, + "gridPos": { + "h": 2, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 31, + "options": { + "content": "

FI*hnbe;6UTx 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�L|kMdK1332 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 b4rM4q8Ol)Zr2KyY0096020Dc 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